mirror of
https://github.com/invoke-ai/InvokeAI.git
synced 2026-01-20 05:18:17 -05:00
Compare commits
323 Commits
v6.5.1
...
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 | ||
|
|
25430f04c5 | ||
|
|
b2b53c4481 | ||
|
|
c6696d7913 | ||
|
|
8bcb6648f1 | ||
|
|
0ee360ba6c | ||
|
|
09bbe3eef9 | ||
|
|
d14b7a48f5 | ||
|
|
1db55b0ffa | ||
|
|
3104a1baa6 | ||
|
|
0e523ca2c1 | ||
|
|
75daef2aba | ||
|
|
b036b18986 | ||
|
|
93535fa3c2 | ||
|
|
dcafb44f8a | ||
|
|
44b1d8d1fc | ||
|
|
6f70a6bd10 | ||
|
|
0546aeed1d | ||
|
|
8933f3f5dd | ||
|
|
29cdefe873 | ||
|
|
df299bb37f | ||
|
|
481fb42371 | ||
|
|
631a04b48c | ||
|
|
547e1941f4 | ||
|
|
031d25ed63 | ||
|
|
27f4af0eb4 | ||
|
|
e0a0617093 | ||
|
|
e6a763b887 | ||
|
|
3c9c49f7d9 | ||
|
|
26690d47b7 | ||
|
|
fcaff6ce09 | ||
|
|
afd7296cb2 | ||
|
|
d6f42c76d5 | ||
|
|
68f39fe907 | ||
|
|
23a528545f | ||
|
|
c69d04a7f0 | ||
|
|
60f1e2d7ad | ||
|
|
cb386bec28 | ||
|
|
f29ceb3f12 | ||
|
|
4f51bc9421 | ||
|
|
0c41abab79 | ||
|
|
cb457c3402 | ||
|
|
606ad73814 | ||
|
|
fe70bd538a | ||
|
|
b5c7316c0a | ||
|
|
460aec03ea | ||
|
|
6730d86a13 | ||
|
|
c4bc03cb1f | ||
|
|
136ee28199 | ||
|
|
2c6d266c0a | ||
|
|
f779920eaa | ||
|
|
01bef5d165 | ||
|
|
72851d3e84 | ||
|
|
4ba85c62ca | ||
|
|
313aedb00a | ||
|
|
85bd324d74 | ||
|
|
4a04411e74 | ||
|
|
299a4db3bb | ||
|
|
390faa592c | ||
|
|
2463aeb84a | ||
|
|
ec8df163d1 | ||
|
|
a198b7da78 | ||
|
|
fb11770852 | ||
|
|
6b6f3d56f7 | ||
|
|
29d00eef9a | ||
|
|
6972cd708d | ||
|
|
82893804ff | ||
|
|
47ffe365bc | ||
|
|
f7b03b1e63 | ||
|
|
356e38e82a | ||
|
|
5ea077bb8c | ||
|
|
3c4b303555 | ||
|
|
b8651cb1a2 | ||
|
|
a6527c0ba1 | ||
|
|
6e40eca754 | ||
|
|
53fab17c33 | ||
|
|
3876d88b3c | ||
|
|
82b4526691 | ||
|
|
f56ba11394 | ||
|
|
32eb5190f2 | ||
|
|
72e378789d | ||
|
|
f10ddb0cab | ||
|
|
286127077d | ||
|
|
36278bc044 | ||
|
|
7a1c7ca43a | ||
|
|
8303d567d5 | ||
|
|
1fe19c1242 | ||
|
|
127a43865c | ||
|
|
24a48884cb | ||
|
|
47cee816fd | ||
|
|
90bacaddda | ||
|
|
c0cc9f421e | ||
|
|
dbb9032648 | ||
|
|
b9e32e59a2 | ||
|
|
545a1d8737 | ||
|
|
c4718403a2 | ||
|
|
eb308b1ff7 | ||
|
|
a277bea804 | ||
|
|
30619c0420 | ||
|
|
504d8e32be | ||
|
|
f21229cd14 | ||
|
|
640ec676c3 | ||
|
|
6370412e9c | ||
|
|
edec2c2775 | ||
|
|
bd38be31d8 | ||
|
|
b938ae0a7e | ||
|
|
6e5b1ed55f | ||
|
|
5970bd38c2 | ||
|
|
e046417cf5 | ||
|
|
27a2cd19bd | ||
|
|
0df631b802 | ||
|
|
5bb7cd168d | ||
|
|
b4ba84ad35 | ||
|
|
d1628f51c9 | ||
|
|
17c1304ce2 | ||
|
|
cc9a85f7d0 | ||
|
|
7e2999649a | ||
|
|
1473142f73 | ||
|
|
49343546e7 | ||
|
|
39d5879405 | ||
|
|
4b4ec29a09 | ||
|
|
dc6811076f | ||
|
|
0568784ee9 | ||
|
|
895eac6bcd | ||
|
|
fe0efa9bdf | ||
|
|
acabc8bd54 | ||
|
|
89f999af08 | ||
|
|
9ae76bef51 | ||
|
|
0999b43616 | ||
|
|
e6e4f58163 | ||
|
|
b371930e02 | ||
|
|
9b50e2303b | ||
|
|
49d1810991 |
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_
|
- [ ] _The PR has a short but descriptive title, suitable for a changelog_
|
||||||
- [ ] _Tests added / updated (if applicable)_
|
- [ ] _Tests added / updated (if applicable)_
|
||||||
|
- [ ] _❗Changes to a redux slice have a corresponding migration_
|
||||||
- [ ] _Documentation added / updated (if applicable)_
|
- [ ] _Documentation added / updated (if applicable)_
|
||||||
- [ ] _Updated `What's New` copy (if doing a release after this PR)_
|
- [ ] _Updated `What's New` copy (if doing a release after this PR)_
|
||||||
|
|||||||
39
invokeai/app/api/routers/board_videos.py
Normal file
39
invokeai/app/api/routers/board_videos.py
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
from fastapi import Body, HTTPException
|
||||||
|
from fastapi.routing import APIRouter
|
||||||
|
|
||||||
|
from invokeai.app.services.videos_common import AddVideosToBoardResult, RemoveVideosFromBoardResult
|
||||||
|
|
||||||
|
board_videos_router = APIRouter(prefix="/v1/board_videos", tags=["boards"])
|
||||||
|
|
||||||
|
|
||||||
|
@board_videos_router.post(
|
||||||
|
"/batch",
|
||||||
|
operation_id="add_videos_to_board",
|
||||||
|
responses={
|
||||||
|
201: {"description": "Videos were added to board successfully"},
|
||||||
|
},
|
||||||
|
status_code=201,
|
||||||
|
response_model=AddVideosToBoardResult,
|
||||||
|
)
|
||||||
|
async def add_videos_to_board(
|
||||||
|
board_id: str = Body(description="The id of the board to add to"),
|
||||||
|
video_ids: list[str] = Body(description="The ids of the videos to add", embed=True),
|
||||||
|
) -> AddVideosToBoardResult:
|
||||||
|
"""Adds a list of videos to a board"""
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@board_videos_router.post(
|
||||||
|
"/batch/delete",
|
||||||
|
operation_id="remove_videos_from_board",
|
||||||
|
responses={
|
||||||
|
201: {"description": "Videos were removed from board successfully"},
|
||||||
|
},
|
||||||
|
status_code=201,
|
||||||
|
response_model=RemoveVideosFromBoardResult,
|
||||||
|
)
|
||||||
|
async def remove_videos_from_board(
|
||||||
|
video_ids: list[str] = Body(description="The ids of the videos to remove", embed=True),
|
||||||
|
) -> RemoveVideosFromBoardResult:
|
||||||
|
"""Removes a list of videos from their board, if they had one"""
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
@@ -7,7 +7,6 @@ from pydantic import BaseModel, Field
|
|||||||
from invokeai.app.api.dependencies import ApiDependencies
|
from invokeai.app.api.dependencies import ApiDependencies
|
||||||
from invokeai.app.services.session_processor.session_processor_common import SessionProcessorStatus
|
from invokeai.app.services.session_processor.session_processor_common import SessionProcessorStatus
|
||||||
from invokeai.app.services.session_queue.session_queue_common import (
|
from invokeai.app.services.session_queue.session_queue_common import (
|
||||||
QUEUE_ITEM_STATUS,
|
|
||||||
Batch,
|
Batch,
|
||||||
BatchStatus,
|
BatchStatus,
|
||||||
CancelAllExceptCurrentResult,
|
CancelAllExceptCurrentResult,
|
||||||
@@ -18,6 +17,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
|||||||
DeleteByDestinationResult,
|
DeleteByDestinationResult,
|
||||||
EnqueueBatchResult,
|
EnqueueBatchResult,
|
||||||
FieldIdentifier,
|
FieldIdentifier,
|
||||||
|
ItemIdsResult,
|
||||||
PruneResult,
|
PruneResult,
|
||||||
RetryItemsResult,
|
RetryItemsResult,
|
||||||
SessionQueueCountsByDestination,
|
SessionQueueCountsByDestination,
|
||||||
@@ -25,7 +25,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
|||||||
SessionQueueItemNotFoundError,
|
SessionQueueItemNotFoundError,
|
||||||
SessionQueueStatus,
|
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"])
|
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}")
|
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(
|
@session_queue_router.get(
|
||||||
"/{queue_id}/list_all",
|
"/{queue_id}/list_all",
|
||||||
operation_id="list_all_queue_items",
|
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}")
|
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(
|
@session_queue_router.put(
|
||||||
"/{queue_id}/processor/resume",
|
"/{queue_id}/processor/resume",
|
||||||
operation_id="resume",
|
operation_id="resume",
|
||||||
@@ -354,7 +374,10 @@ async def get_queue_item(
|
|||||||
) -> SessionQueueItem:
|
) -> SessionQueueItem:
|
||||||
"""Gets a queue item"""
|
"""Gets a queue item"""
|
||||||
try:
|
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:
|
except SessionQueueItemNotFoundError:
|
||||||
raise HTTPException(status_code=404, detail=f"Queue item with id {item_id} not found in queue {queue_id}")
|
raise HTTPException(status_code=404, detail=f"Queue item with id {item_id} not found in queue {queue_id}")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|||||||
119
invokeai/app/api/routers/videos.py
Normal file
119
invokeai/app/api/routers/videos.py
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from fastapi import Body, HTTPException, Path, Query
|
||||||
|
from fastapi.routing import APIRouter
|
||||||
|
|
||||||
|
from invokeai.app.services.shared.pagination import OffsetPaginatedResults
|
||||||
|
from invokeai.app.services.shared.sqlite.sqlite_common import SQLiteDirection
|
||||||
|
from invokeai.app.services.videos_common import (
|
||||||
|
DeleteVideosResult,
|
||||||
|
StarredVideosResult,
|
||||||
|
UnstarredVideosResult,
|
||||||
|
VideoDTO,
|
||||||
|
VideoIdsResult,
|
||||||
|
VideoRecordChanges,
|
||||||
|
)
|
||||||
|
|
||||||
|
videos_router = APIRouter(prefix="/v1/videos", tags=["videos"])
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.patch(
|
||||||
|
"/i/{video_id}",
|
||||||
|
operation_id="update_video",
|
||||||
|
response_model=VideoDTO,
|
||||||
|
)
|
||||||
|
async def update_video(
|
||||||
|
video_id: str = Path(description="The id of the video to update"),
|
||||||
|
video_changes: VideoRecordChanges = Body(description="The changes to apply to the video"),
|
||||||
|
) -> VideoDTO:
|
||||||
|
"""Updates a video"""
|
||||||
|
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.get(
|
||||||
|
"/i/{video_id}",
|
||||||
|
operation_id="get_video_dto",
|
||||||
|
response_model=VideoDTO,
|
||||||
|
)
|
||||||
|
async def get_video_dto(
|
||||||
|
video_id: str = Path(description="The id of the video to get"),
|
||||||
|
) -> VideoDTO:
|
||||||
|
"""Gets a video's DTO"""
|
||||||
|
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.post("/delete", operation_id="delete_videos_from_list", response_model=DeleteVideosResult)
|
||||||
|
async def delete_videos_from_list(
|
||||||
|
video_ids: list[str] = Body(description="The list of ids of videos to delete", embed=True),
|
||||||
|
) -> DeleteVideosResult:
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.post("/star", operation_id="star_videos_in_list", response_model=StarredVideosResult)
|
||||||
|
async def star_videos_in_list(
|
||||||
|
video_ids: list[str] = Body(description="The list of ids of videos to star", embed=True),
|
||||||
|
) -> StarredVideosResult:
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.post("/unstar", operation_id="unstar_videos_in_list", response_model=UnstarredVideosResult)
|
||||||
|
async def unstar_videos_in_list(
|
||||||
|
video_ids: list[str] = Body(description="The list of ids of videos to unstar", embed=True),
|
||||||
|
) -> UnstarredVideosResult:
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.delete("/uncategorized", operation_id="delete_uncategorized_videos", response_model=DeleteVideosResult)
|
||||||
|
async def delete_uncategorized_videos() -> DeleteVideosResult:
|
||||||
|
"""Deletes all videos that are uncategorized"""
|
||||||
|
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.get("/", operation_id="list_video_dtos", response_model=OffsetPaginatedResults[VideoDTO])
|
||||||
|
async def list_video_dtos(
|
||||||
|
is_intermediate: Optional[bool] = Query(default=None, description="Whether to list intermediate videos."),
|
||||||
|
board_id: Optional[str] = Query(
|
||||||
|
default=None,
|
||||||
|
description="The board id to filter by. Use 'none' to find videos without a board.",
|
||||||
|
),
|
||||||
|
offset: int = Query(default=0, description="The page offset"),
|
||||||
|
limit: int = Query(default=10, description="The number of videos per page"),
|
||||||
|
order_dir: SQLiteDirection = Query(default=SQLiteDirection.Descending, description="The order of sort"),
|
||||||
|
starred_first: bool = Query(default=True, description="Whether to sort by starred videos first"),
|
||||||
|
search_term: Optional[str] = Query(default=None, description="The term to search for"),
|
||||||
|
) -> OffsetPaginatedResults[VideoDTO]:
|
||||||
|
"""Lists video DTOs"""
|
||||||
|
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.get("/ids", operation_id="get_video_ids")
|
||||||
|
async def get_video_ids(
|
||||||
|
is_intermediate: Optional[bool] = Query(default=None, description="Whether to list intermediate videos."),
|
||||||
|
board_id: Optional[str] = Query(
|
||||||
|
default=None,
|
||||||
|
description="The board id to filter by. Use 'none' to find videos without a board.",
|
||||||
|
),
|
||||||
|
order_dir: SQLiteDirection = Query(default=SQLiteDirection.Descending, description="The order of sort"),
|
||||||
|
starred_first: bool = Query(default=True, description="Whether to sort by starred videos first"),
|
||||||
|
search_term: Optional[str] = Query(default=None, description="The term to search for"),
|
||||||
|
) -> VideoIdsResult:
|
||||||
|
"""Gets ordered list of video ids with metadata for optimistic updates"""
|
||||||
|
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
|
|
||||||
|
|
||||||
|
@videos_router.post(
|
||||||
|
"/videos_by_ids",
|
||||||
|
operation_id="get_videos_by_ids",
|
||||||
|
responses={200: {"model": list[VideoDTO]}},
|
||||||
|
)
|
||||||
|
async def get_videos_by_ids(
|
||||||
|
video_ids: list[str] = Body(embed=True, description="Object containing list of video ids to fetch DTOs for"),
|
||||||
|
) -> list[VideoDTO]:
|
||||||
|
"""Gets video DTOs for the specified video ids. Maintains order of input ids."""
|
||||||
|
|
||||||
|
raise HTTPException(status_code=501, detail="Not implemented")
|
||||||
@@ -18,6 +18,7 @@ from invokeai.app.api.no_cache_staticfiles import NoCacheStaticFiles
|
|||||||
from invokeai.app.api.routers import (
|
from invokeai.app.api.routers import (
|
||||||
app_info,
|
app_info,
|
||||||
board_images,
|
board_images,
|
||||||
|
board_videos,
|
||||||
boards,
|
boards,
|
||||||
client_state,
|
client_state,
|
||||||
download_queue,
|
download_queue,
|
||||||
@@ -27,6 +28,7 @@ from invokeai.app.api.routers import (
|
|||||||
session_queue,
|
session_queue,
|
||||||
style_presets,
|
style_presets,
|
||||||
utilities,
|
utilities,
|
||||||
|
videos,
|
||||||
workflows,
|
workflows,
|
||||||
)
|
)
|
||||||
from invokeai.app.api.sockets import SocketIO
|
from invokeai.app.api.sockets import SocketIO
|
||||||
@@ -125,8 +127,10 @@ app.include_router(utilities.utilities_router, prefix="/api")
|
|||||||
app.include_router(model_manager.model_manager_router, prefix="/api")
|
app.include_router(model_manager.model_manager_router, prefix="/api")
|
||||||
app.include_router(download_queue.download_queue_router, prefix="/api")
|
app.include_router(download_queue.download_queue_router, prefix="/api")
|
||||||
app.include_router(images.images_router, prefix="/api")
|
app.include_router(images.images_router, prefix="/api")
|
||||||
|
app.include_router(videos.videos_router, prefix="/api")
|
||||||
app.include_router(boards.boards_router, prefix="/api")
|
app.include_router(boards.boards_router, prefix="/api")
|
||||||
app.include_router(board_images.board_images_router, prefix="/api")
|
app.include_router(board_images.board_images_router, prefix="/api")
|
||||||
|
app.include_router(board_videos.board_videos_router, prefix="/api")
|
||||||
app.include_router(model_relationships.model_relationships_router, prefix="/api")
|
app.include_router(model_relationships.model_relationships_router, prefix="/api")
|
||||||
app.include_router(app_info.app_router, prefix="/api")
|
app.include_router(app_info.app_router, prefix="/api")
|
||||||
app.include_router(session_queue.session_queue_router, prefix="/api")
|
app.include_router(session_queue.session_queue_router, prefix="/api")
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
from enum import Enum
|
from enum import Enum
|
||||||
from typing import Any, Callable, Optional, Tuple
|
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.fields import _Unset
|
||||||
from pydantic_core import PydanticUndefined
|
from pydantic_core import PydanticUndefined
|
||||||
|
|
||||||
from invokeai.app.util.metaenum import MetaEnum
|
from invokeai.app.util.metaenum import MetaEnum
|
||||||
|
from invokeai.backend.image_util.segment_anything.shared import BoundingBox
|
||||||
from invokeai.backend.util.logging import InvokeAILogger
|
from invokeai.backend.util.logging import InvokeAILogger
|
||||||
|
|
||||||
logger = InvokeAILogger.get_logger()
|
logger = InvokeAILogger.get_logger()
|
||||||
@@ -66,11 +67,14 @@ class UIType(str, Enum, metaclass=MetaEnum):
|
|||||||
ChatGPT4oModel = "ChatGPT4oModelField"
|
ChatGPT4oModel = "ChatGPT4oModelField"
|
||||||
Gemini2_5Model = "Gemini2_5ModelField"
|
Gemini2_5Model = "Gemini2_5ModelField"
|
||||||
FluxKontextModel = "FluxKontextModelField"
|
FluxKontextModel = "FluxKontextModelField"
|
||||||
|
Veo3Model = "Veo3ModelField"
|
||||||
|
RunwayModel = "RunwayModelField"
|
||||||
# endregion
|
# endregion
|
||||||
|
|
||||||
# region Misc Field Types
|
# region Misc Field Types
|
||||||
Scheduler = "SchedulerField"
|
Scheduler = "SchedulerField"
|
||||||
Any = "AnyField"
|
Any = "AnyField"
|
||||||
|
Video = "VideoField"
|
||||||
# endregion
|
# endregion
|
||||||
|
|
||||||
# region Internal Field Types
|
# region Internal Field Types
|
||||||
@@ -225,6 +229,12 @@ class ImageField(BaseModel):
|
|||||||
image_name: str = Field(description="The name of the image")
|
image_name: str = Field(description="The name of the image")
|
||||||
|
|
||||||
|
|
||||||
|
class VideoField(BaseModel):
|
||||||
|
"""A video primitive field"""
|
||||||
|
|
||||||
|
video_id: str = Field(description="The id of the video")
|
||||||
|
|
||||||
|
|
||||||
class BoardField(BaseModel):
|
class BoardField(BaseModel):
|
||||||
"""A board primitive field"""
|
"""A board primitive field"""
|
||||||
|
|
||||||
@@ -322,14 +332,9 @@ class ConditioningField(BaseModel):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class BoundingBoxField(BaseModel):
|
class BoundingBoxField(BoundingBox):
|
||||||
"""A bounding box primitive value."""
|
"""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(
|
score: Optional[float] = Field(
|
||||||
default=None,
|
default=None,
|
||||||
ge=0.0,
|
ge=0.0,
|
||||||
@@ -338,21 +343,6 @@ class BoundingBoxField(BaseModel):
|
|||||||
"when the bounding box was produced by a detector and has an associated confidence score.",
|
"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]]):
|
class MetadataField(RootModel[dict[str, Any]]):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -27,6 +27,7 @@ from invokeai.app.invocations.fields import (
|
|||||||
SD3ConditioningField,
|
SD3ConditioningField,
|
||||||
TensorField,
|
TensorField,
|
||||||
UIComponent,
|
UIComponent,
|
||||||
|
VideoField,
|
||||||
)
|
)
|
||||||
from invokeai.app.services.images.images_common import ImageDTO
|
from invokeai.app.services.images.images_common import ImageDTO
|
||||||
from invokeai.app.services.shared.invocation_context import InvocationContext
|
from invokeai.app.services.shared.invocation_context import InvocationContext
|
||||||
@@ -287,6 +288,30 @@ class ImageCollectionInvocation(BaseInvocation):
|
|||||||
return ImageCollectionOutput(collection=self.collection)
|
return ImageCollectionOutput(collection=self.collection)
|
||||||
|
|
||||||
|
|
||||||
|
# endregion
|
||||||
|
|
||||||
|
# region Video
|
||||||
|
|
||||||
|
|
||||||
|
@invocation_output("video_output")
|
||||||
|
class VideoOutput(BaseInvocationOutput):
|
||||||
|
"""Base class for nodes that output a video"""
|
||||||
|
|
||||||
|
video: VideoField = OutputField(description="The output video")
|
||||||
|
width: int = OutputField(description="The width of the video in pixels")
|
||||||
|
height: int = OutputField(description="The height of the video in pixels")
|
||||||
|
duration_seconds: float = OutputField(description="The duration of the video in seconds")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def build(cls, video_id: str, width: int, height: int, duration_seconds: float) -> "VideoOutput":
|
||||||
|
return cls(
|
||||||
|
video=VideoField(video_id=video_id),
|
||||||
|
width=width,
|
||||||
|
height=height,
|
||||||
|
duration_seconds=duration_seconds,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
# endregion
|
# endregion
|
||||||
|
|
||||||
# region DenoiseMask
|
# region DenoiseMask
|
||||||
|
|||||||
@@ -1,72 +1,75 @@
|
|||||||
from enum import Enum
|
from itertools import zip_longest
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Literal
|
from typing import Literal
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
import torch
|
import torch
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
from pydantic import BaseModel, Field
|
from pydantic import BaseModel, Field, model_validator
|
||||||
from transformers import AutoProcessor
|
|
||||||
from transformers.models.sam import SamModel
|
from transformers.models.sam import SamModel
|
||||||
from transformers.models.sam.processing_sam import SamProcessor
|
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.baseinvocation import BaseInvocation, invocation
|
||||||
from invokeai.app.invocations.fields import BoundingBoxField, ImageField, InputField, TensorField
|
from invokeai.app.invocations.fields import BoundingBoxField, ImageField, InputField, TensorField
|
||||||
from invokeai.app.invocations.primitives import MaskOutput
|
from invokeai.app.invocations.primitives import MaskOutput
|
||||||
from invokeai.app.services.shared.invocation_context import InvocationContext
|
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.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.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_MODEL_IDS: dict[SegmentAnythingModelKey, str] = {
|
||||||
"segment-anything-base": "facebook/sam-vit-base",
|
"segment-anything-base": "facebook/sam-vit-base",
|
||||||
"segment-anything-large": "facebook/sam-vit-large",
|
"segment-anything-large": "facebook/sam-vit-large",
|
||||||
"segment-anything-huge": "facebook/sam-vit-huge",
|
"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):
|
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]
|
return [[point.x, point.y, point.label.value] for point in self.points]
|
||||||
|
|
||||||
|
|
||||||
@invocation(
|
@invocation(
|
||||||
"segment_anything",
|
"segment_anything",
|
||||||
title="Segment Anything",
|
title="Segment Anything",
|
||||||
tags=["prompt", "segmentation"],
|
tags=["prompt", "segmentation", "sam", "sam2"],
|
||||||
category="segmentation",
|
category="segmentation",
|
||||||
version="1.2.0",
|
version="1.3.0",
|
||||||
)
|
)
|
||||||
class SegmentAnythingInvocation(BaseInvocation):
|
class SegmentAnythingInvocation(BaseInvocation):
|
||||||
"""Runs a Segment Anything Model."""
|
"""Runs a Segment Anything Model (SAM or SAM2)."""
|
||||||
|
|
||||||
# Reference:
|
# Reference:
|
||||||
# - https://arxiv.org/pdf/2304.02643
|
# - https://arxiv.org/pdf/2304.02643
|
||||||
# - https://huggingface.co/docs/transformers/v4.43.3/en/model_doc/grounding-dino#grounded-sam
|
# - 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
|
# - 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.")
|
image: ImageField = InputField(description="The image to segment.")
|
||||||
bounding_boxes: list[BoundingBoxField] | None = InputField(
|
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(
|
point_lists: list[SAMPointsField] | None = InputField(
|
||||||
default=None,
|
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(
|
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).",
|
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",
|
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()
|
@torch.no_grad()
|
||||||
def invoke(self, context: InvocationContext) -> MaskOutput:
|
def invoke(self, context: InvocationContext) -> MaskOutput:
|
||||||
# The models expect a 3-channel RGB image.
|
# The models expect a 3-channel RGB image.
|
||||||
image_pil = context.images.get_pil(self.image.image_name, mode="RGB")
|
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 (
|
if (not self.bounding_boxes or len(self.bounding_boxes) == 0) and (
|
||||||
not self.point_lists or len(self.point_lists) == 0
|
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.
|
# model, and figure out how to make it work in the pipeline.
|
||||||
# torch_dtype=TorchDevice.choose_torch_dtype(),
|
# torch_dtype=TorchDevice.choose_torch_dtype(),
|
||||||
)
|
)
|
||||||
|
sam_processor = SamProcessor.from_pretrained(model_path, local_files_only=True)
|
||||||
sam_processor = AutoProcessor.from_pretrained(model_path, local_files_only=True)
|
|
||||||
assert isinstance(sam_processor, SamProcessor)
|
|
||||||
return SegmentAnythingPipeline(sam_model=sam_model, sam_processor=sam_processor)
|
return SegmentAnythingPipeline(sam_model=sam_model, sam_processor=sam_processor)
|
||||||
|
|
||||||
def _segment(self, context: InvocationContext, image: Image.Image) -> list[torch.Tensor]:
|
@staticmethod
|
||||||
"""Use Segment Anything (SAM) to generate masks given an image + a set of bounding boxes."""
|
def _load_sam_2_model(model_path: Path):
|
||||||
# Convert the bounding boxes to the SAM input format.
|
sam2_model = Sam2Model.from_pretrained(model_path, local_files_only=True)
|
||||||
sam_bounding_boxes = (
|
sam2_processor = Sam2Processor.from_pretrained(model_path, local_files_only=True)
|
||||||
[[bb.x_min, bb.y_min, bb.x_max, bb.y_max] for bb in self.bounding_boxes] if self.bounding_boxes else None
|
return SegmentAnything2Pipeline(sam2_model=sam2_model, sam2_processor=sam2_processor)
|
||||||
)
|
|
||||||
sam_points = [p.to_list() for p in self.point_lists] if self.point_lists else None
|
|
||||||
|
|
||||||
with (
|
def _segment(self, context: InvocationContext, image: Image.Image) -> list[torch.Tensor]:
|
||||||
context.models.load_remote_model(
|
"""Use Segment Anything (SAM or SAM2) to generate masks given an image + a set of bounding boxes."""
|
||||||
source=SEGMENT_ANYTHING_MODEL_IDS[self.model], loader=SegmentAnythingInvocation._load_sam_model
|
|
||||||
) as sam_pipeline,
|
source = SEGMENT_ANYTHING_MODEL_IDS[self.model]
|
||||||
):
|
inputs: list[SAMInput] = []
|
||||||
assert isinstance(sam_pipeline, SegmentAnythingPipeline)
|
for bbox_field, point_field in zip_longest(self.bounding_boxes or [], self.point_lists or [], fillvalue=None):
|
||||||
masks = sam_pipeline.segment(image=image, bounding_boxes=sam_bounding_boxes, point_lists=sam_points)
|
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)
|
masks = self._process_masks(masks)
|
||||||
if self.apply_polygon_refinement:
|
if self.apply_polygon_refinement:
|
||||||
|
|||||||
@@ -49,3 +49,11 @@ class BoardImageRecordStorageBase(ABC):
|
|||||||
) -> int:
|
) -> int:
|
||||||
"""Gets the number of images for a board."""
|
"""Gets the number of images for a board."""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_asset_count_for_board(
|
||||||
|
self,
|
||||||
|
board_id: str,
|
||||||
|
) -> int:
|
||||||
|
"""Gets the number of assets for a board."""
|
||||||
|
pass
|
||||||
|
|||||||
@@ -3,6 +3,8 @@ from typing import Optional, cast
|
|||||||
|
|
||||||
from invokeai.app.services.board_image_records.board_image_records_base import BoardImageRecordStorageBase
|
from invokeai.app.services.board_image_records.board_image_records_base import BoardImageRecordStorageBase
|
||||||
from invokeai.app.services.image_records.image_records_common import (
|
from invokeai.app.services.image_records.image_records_common import (
|
||||||
|
ASSETS_CATEGORIES,
|
||||||
|
IMAGE_CATEGORIES,
|
||||||
ImageCategory,
|
ImageCategory,
|
||||||
ImageRecord,
|
ImageRecord,
|
||||||
deserialize_image_record,
|
deserialize_image_record,
|
||||||
@@ -151,15 +153,38 @@ class SqliteBoardImageRecordStorage(BoardImageRecordStorageBase):
|
|||||||
|
|
||||||
def get_image_count_for_board(self, board_id: str) -> int:
|
def get_image_count_for_board(self, board_id: str) -> int:
|
||||||
with self._db.transaction() as cursor:
|
with self._db.transaction() as cursor:
|
||||||
|
# Convert the enum values to unique list of strings
|
||||||
|
category_strings = [c.value for c in set(IMAGE_CATEGORIES)]
|
||||||
|
# Create the correct length of placeholders
|
||||||
|
placeholders = ",".join("?" * len(category_strings))
|
||||||
cursor.execute(
|
cursor.execute(
|
||||||
"""--sql
|
f"""--sql
|
||||||
SELECT COUNT(*)
|
SELECT COUNT(*)
|
||||||
FROM board_images
|
FROM board_images
|
||||||
INNER JOIN images ON board_images.image_name = images.image_name
|
INNER JOIN images ON board_images.image_name = images.image_name
|
||||||
WHERE images.is_intermediate = FALSE
|
WHERE images.is_intermediate = FALSE AND images.image_category IN ( {placeholders} )
|
||||||
AND board_images.board_id = ?;
|
AND board_images.board_id = ?;
|
||||||
""",
|
""",
|
||||||
(board_id,),
|
(*category_strings, board_id),
|
||||||
|
)
|
||||||
|
count = cast(int, cursor.fetchone()[0])
|
||||||
|
return count
|
||||||
|
|
||||||
|
def get_asset_count_for_board(self, board_id: str) -> int:
|
||||||
|
with self._db.transaction() as cursor:
|
||||||
|
# Convert the enum values to unique list of strings
|
||||||
|
category_strings = [c.value for c in set(ASSETS_CATEGORIES)]
|
||||||
|
# Create the correct length of placeholders
|
||||||
|
placeholders = ",".join("?" * len(category_strings))
|
||||||
|
cursor.execute(
|
||||||
|
f"""--sql
|
||||||
|
SELECT COUNT(*)
|
||||||
|
FROM board_images
|
||||||
|
INNER JOIN images ON board_images.image_name = images.image_name
|
||||||
|
WHERE images.is_intermediate = FALSE AND images.image_category IN ( {placeholders} )
|
||||||
|
AND board_images.board_id = ?;
|
||||||
|
""",
|
||||||
|
(*category_strings, board_id),
|
||||||
)
|
)
|
||||||
count = cast(int, cursor.fetchone()[0])
|
count = cast(int, cursor.fetchone()[0])
|
||||||
return count
|
return count
|
||||||
|
|||||||
@@ -12,12 +12,20 @@ class BoardDTO(BoardRecord):
|
|||||||
"""The URL of the thumbnail of the most recent image in the board."""
|
"""The URL of the thumbnail of the most recent image in the board."""
|
||||||
image_count: int = Field(description="The number of images in the board.")
|
image_count: int = Field(description="The number of images in the board.")
|
||||||
"""The number of images in the board."""
|
"""The number of images in the board."""
|
||||||
|
asset_count: int = Field(description="The number of assets in the board.")
|
||||||
|
"""The number of assets in the board."""
|
||||||
|
video_count: int = Field(description="The number of videos in the board.")
|
||||||
|
"""The number of videos in the board."""
|
||||||
|
|
||||||
|
|
||||||
def board_record_to_dto(board_record: BoardRecord, cover_image_name: Optional[str], image_count: int) -> BoardDTO:
|
def board_record_to_dto(
|
||||||
|
board_record: BoardRecord, cover_image_name: Optional[str], image_count: int, asset_count: int, video_count: int
|
||||||
|
) -> BoardDTO:
|
||||||
"""Converts a board record to a board DTO."""
|
"""Converts a board record to a board DTO."""
|
||||||
return BoardDTO(
|
return BoardDTO(
|
||||||
**board_record.model_dump(exclude={"cover_image_name"}),
|
**board_record.model_dump(exclude={"cover_image_name"}),
|
||||||
cover_image_name=cover_image_name,
|
cover_image_name=cover_image_name,
|
||||||
image_count=image_count,
|
image_count=image_count,
|
||||||
|
asset_count=asset_count,
|
||||||
|
video_count=video_count,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ class BoardService(BoardServiceABC):
|
|||||||
board_name: str,
|
board_name: str,
|
||||||
) -> BoardDTO:
|
) -> BoardDTO:
|
||||||
board_record = self.__invoker.services.board_records.save(board_name)
|
board_record = self.__invoker.services.board_records.save(board_name)
|
||||||
return board_record_to_dto(board_record, None, 0)
|
return board_record_to_dto(board_record, None, 0, 0, 0)
|
||||||
|
|
||||||
def get_dto(self, board_id: str) -> BoardDTO:
|
def get_dto(self, board_id: str) -> BoardDTO:
|
||||||
board_record = self.__invoker.services.board_records.get(board_id)
|
board_record = self.__invoker.services.board_records.get(board_id)
|
||||||
@@ -27,7 +27,9 @@ class BoardService(BoardServiceABC):
|
|||||||
else:
|
else:
|
||||||
cover_image_name = None
|
cover_image_name = None
|
||||||
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(board_id)
|
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(board_id)
|
||||||
return board_record_to_dto(board_record, cover_image_name, image_count)
|
asset_count = self.__invoker.services.board_image_records.get_asset_count_for_board(board_id)
|
||||||
|
video_count = 0 # noop for OSS
|
||||||
|
return board_record_to_dto(board_record, cover_image_name, image_count, asset_count, video_count)
|
||||||
|
|
||||||
def update(
|
def update(
|
||||||
self,
|
self,
|
||||||
@@ -42,7 +44,9 @@ class BoardService(BoardServiceABC):
|
|||||||
cover_image_name = None
|
cover_image_name = None
|
||||||
|
|
||||||
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(board_id)
|
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(board_id)
|
||||||
return board_record_to_dto(board_record, cover_image_name, image_count)
|
asset_count = self.__invoker.services.board_image_records.get_asset_count_for_board(board_id)
|
||||||
|
video_count = 0 # noop for OSS
|
||||||
|
return board_record_to_dto(board_record, cover_image_name, image_count, asset_count, video_count)
|
||||||
|
|
||||||
def delete(self, board_id: str) -> None:
|
def delete(self, board_id: str) -> None:
|
||||||
self.__invoker.services.board_records.delete(board_id)
|
self.__invoker.services.board_records.delete(board_id)
|
||||||
@@ -67,7 +71,9 @@ class BoardService(BoardServiceABC):
|
|||||||
cover_image_name = None
|
cover_image_name = None
|
||||||
|
|
||||||
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(r.board_id)
|
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(r.board_id)
|
||||||
board_dtos.append(board_record_to_dto(r, cover_image_name, image_count))
|
asset_count = self.__invoker.services.board_image_records.get_asset_count_for_board(r.board_id)
|
||||||
|
video_count = 0 # noop for OSS
|
||||||
|
board_dtos.append(board_record_to_dto(r, cover_image_name, image_count, asset_count, video_count))
|
||||||
|
|
||||||
return OffsetPaginatedResults[BoardDTO](items=board_dtos, offset=offset, limit=limit, total=len(board_dtos))
|
return OffsetPaginatedResults[BoardDTO](items=board_dtos, offset=offset, limit=limit, total=len(board_dtos))
|
||||||
|
|
||||||
@@ -84,6 +90,8 @@ class BoardService(BoardServiceABC):
|
|||||||
cover_image_name = None
|
cover_image_name = None
|
||||||
|
|
||||||
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(r.board_id)
|
image_count = self.__invoker.services.board_image_records.get_image_count_for_board(r.board_id)
|
||||||
board_dtos.append(board_record_to_dto(r, cover_image_name, image_count))
|
asset_count = self.__invoker.services.board_image_records.get_asset_count_for_board(r.board_id)
|
||||||
|
video_count = 0 # noop for OSS
|
||||||
|
board_dtos.append(board_record_to_dto(r, cover_image_name, image_count, asset_count, video_count))
|
||||||
|
|
||||||
return board_dtos
|
return board_dtos
|
||||||
|
|||||||
@@ -150,4 +150,15 @@ class BulkDownloadService(BulkDownloadBase):
|
|||||||
def _is_valid_path(self, path: Union[str, Path]) -> bool:
|
def _is_valid_path(self, path: Union[str, Path]) -> bool:
|
||||||
"""Validates the path given for a bulk download."""
|
"""Validates the path given for a bulk download."""
|
||||||
path = path if isinstance(path, Path) else Path(path)
|
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_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_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")
|
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")
|
created_at: str = Field(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")
|
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")
|
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")
|
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")
|
batch_status: BatchStatus = Field(description="The status of the batch")
|
||||||
@@ -258,8 +258,8 @@ class QueueItemStatusChangedEvent(QueueItemEventBase):
|
|||||||
error_type=queue_item.error_type,
|
error_type=queue_item.error_type,
|
||||||
error_message=queue_item.error_message,
|
error_message=queue_item.error_message,
|
||||||
error_traceback=queue_item.error_traceback,
|
error_traceback=queue_item.error_traceback,
|
||||||
created_at=str(queue_item.created_at) if queue_item.created_at else None,
|
created_at=str(queue_item.created_at),
|
||||||
updated_at=str(queue_item.updated_at) if queue_item.updated_at else None,
|
updated_at=str(queue_item.updated_at),
|
||||||
started_at=str(queue_item.started_at) if queue_item.started_at else None,
|
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,
|
completed_at=str(queue_item.completed_at) if queue_item.completed_at else None,
|
||||||
batch_status=batch_status,
|
batch_status=batch_status,
|
||||||
|
|||||||
@@ -58,6 +58,15 @@ class ImageCategory(str, Enum, metaclass=MetaEnum):
|
|||||||
"""OTHER: The image is some other type of image with a specialized purpose. To be used by external nodes."""
|
"""OTHER: The image is some other type of image with a specialized purpose. To be used by external nodes."""
|
||||||
|
|
||||||
|
|
||||||
|
IMAGE_CATEGORIES: list[ImageCategory] = [ImageCategory.GENERAL]
|
||||||
|
ASSETS_CATEGORIES: list[ImageCategory] = [
|
||||||
|
ImageCategory.CONTROL,
|
||||||
|
ImageCategory.MASK,
|
||||||
|
ImageCategory.USER,
|
||||||
|
ImageCategory.OTHER,
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
class InvalidImageCategoryException(ValueError):
|
class InvalidImageCategoryException(ValueError):
|
||||||
"""Raised when a provided value is not a valid ImageCategory.
|
"""Raised when a provided value is not a valid ImageCategory.
|
||||||
|
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ from invokeai.app.util.model_exclude_null import BaseModelExcludeNull
|
|||||||
from invokeai.backend.model_manager.config import (
|
from invokeai.backend.model_manager.config import (
|
||||||
AnyModelConfig,
|
AnyModelConfig,
|
||||||
ControlAdapterDefaultSettings,
|
ControlAdapterDefaultSettings,
|
||||||
|
LoraModelDefaultSettings,
|
||||||
MainModelDefaultSettings,
|
MainModelDefaultSettings,
|
||||||
)
|
)
|
||||||
from invokeai.backend.model_manager.taxonomy import (
|
from invokeai.backend.model_manager.taxonomy import (
|
||||||
@@ -83,8 +84,8 @@ class ModelRecordChanges(BaseModelExcludeNull):
|
|||||||
file_size: Optional[int] = Field(description="Size of model file", default=None)
|
file_size: Optional[int] = Field(description="Size of model file", default=None)
|
||||||
format: Optional[str] = Field(description="format of model file", default=None)
|
format: Optional[str] = Field(description="format of model file", default=None)
|
||||||
trigger_phrases: Optional[set[str]] = Field(description="Set of trigger phrases for this model", default=None)
|
trigger_phrases: Optional[set[str]] = Field(description="Set of trigger phrases for this model", default=None)
|
||||||
default_settings: Optional[MainModelDefaultSettings | ControlAdapterDefaultSettings] = Field(
|
default_settings: Optional[MainModelDefaultSettings | LoraModelDefaultSettings | ControlAdapterDefaultSettings] = (
|
||||||
description="Default settings for this model", default=None
|
Field(description="Default settings for this model", default=None)
|
||||||
)
|
)
|
||||||
|
|
||||||
# Checkpoint-specific changes
|
# Checkpoint-specific changes
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ from abc import ABC, abstractmethod
|
|||||||
from typing import Any, Coroutine, Optional
|
from typing import Any, Coroutine, Optional
|
||||||
|
|
||||||
from invokeai.app.services.session_queue.session_queue_common import (
|
from invokeai.app.services.session_queue.session_queue_common import (
|
||||||
QUEUE_ITEM_STATUS,
|
|
||||||
Batch,
|
Batch,
|
||||||
BatchStatus,
|
BatchStatus,
|
||||||
CancelAllExceptCurrentResult,
|
CancelAllExceptCurrentResult,
|
||||||
@@ -15,6 +14,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
|||||||
EnqueueBatchResult,
|
EnqueueBatchResult,
|
||||||
IsEmptyResult,
|
IsEmptyResult,
|
||||||
IsFullResult,
|
IsFullResult,
|
||||||
|
ItemIdsResult,
|
||||||
PruneResult,
|
PruneResult,
|
||||||
RetryItemsResult,
|
RetryItemsResult,
|
||||||
SessionQueueCountsByDestination,
|
SessionQueueCountsByDestination,
|
||||||
@@ -22,7 +22,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
|||||||
SessionQueueStatus,
|
SessionQueueStatus,
|
||||||
)
|
)
|
||||||
from invokeai.app.services.shared.graph import GraphExecutionState
|
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):
|
class SessionQueueBase(ABC):
|
||||||
@@ -135,19 +135,6 @@ class SessionQueueBase(ABC):
|
|||||||
"""Deletes all queue items except in-progress items"""
|
"""Deletes all queue items except in-progress items"""
|
||||||
pass
|
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
|
@abstractmethod
|
||||||
def list_all_queue_items(
|
def list_all_queue_items(
|
||||||
self,
|
self,
|
||||||
@@ -157,9 +144,18 @@ class SessionQueueBase(ABC):
|
|||||||
"""Gets all queue items that match the given parameters"""
|
"""Gets all queue items that match the given parameters"""
|
||||||
pass
|
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
|
@abstractmethod
|
||||||
def get_queue_item(self, item_id: int) -> SessionQueueItem:
|
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
|
pass
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
|
|||||||
@@ -176,6 +176,14 @@ DEFAULT_QUEUE_ID = "default"
|
|||||||
|
|
||||||
QUEUE_ITEM_STATUS = Literal["pending", "in_progress", "completed", "failed", "canceled"]
|
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])
|
NodeFieldValueValidator = TypeAdapter(list[NodeFieldValue])
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
|||||||
EnqueueBatchResult,
|
EnqueueBatchResult,
|
||||||
IsEmptyResult,
|
IsEmptyResult,
|
||||||
IsFullResult,
|
IsFullResult,
|
||||||
|
ItemIdsResult,
|
||||||
PruneResult,
|
PruneResult,
|
||||||
RetryItemsResult,
|
RetryItemsResult,
|
||||||
SessionQueueCountsByDestination,
|
SessionQueueCountsByDestination,
|
||||||
@@ -33,7 +34,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
|||||||
prepare_values_to_insert,
|
prepare_values_to_insert,
|
||||||
)
|
)
|
||||||
from invokeai.app.services.shared.graph import GraphExecutionState
|
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
|
from invokeai.app.services.shared.sqlite.sqlite_database import SqliteDatabase
|
||||||
|
|
||||||
|
|
||||||
@@ -587,59 +588,6 @@ class SqliteSessionQueue(SessionQueueBase):
|
|||||||
)
|
)
|
||||||
return self.get_queue_item(item_id)
|
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(
|
def list_all_queue_items(
|
||||||
self,
|
self,
|
||||||
queue_id: str,
|
queue_id: str,
|
||||||
@@ -671,6 +619,26 @@ class SqliteSessionQueue(SessionQueueBase):
|
|||||||
items = [SessionQueueItem.queue_item_from_dict(dict(result)) for result in results]
|
items = [SessionQueueItem.queue_item_from_dict(dict(result)) for result in results]
|
||||||
return items
|
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:
|
def get_queue_status(self, queue_id: str) -> SessionQueueStatus:
|
||||||
with self._db.transaction() as cursor:
|
with self._db.transaction() as cursor:
|
||||||
cursor.execute(
|
cursor.execute(
|
||||||
|
|||||||
179
invokeai/app/services/videos_common.py
Normal file
179
invokeai/app/services/videos_common.py
Normal file
@@ -0,0 +1,179 @@
|
|||||||
|
import datetime
|
||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
from pydantic import BaseModel, Field, StrictBool, StrictStr
|
||||||
|
|
||||||
|
from invokeai.app.util.misc import get_iso_timestamp
|
||||||
|
from invokeai.app.util.model_exclude_null import BaseModelExcludeNull
|
||||||
|
|
||||||
|
VIDEO_DTO_COLS = ", ".join(
|
||||||
|
[
|
||||||
|
"videos." + c
|
||||||
|
for c in [
|
||||||
|
"video_id",
|
||||||
|
"width",
|
||||||
|
"height",
|
||||||
|
"session_id",
|
||||||
|
"node_id",
|
||||||
|
"is_intermediate",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
"deleted_at",
|
||||||
|
"starred",
|
||||||
|
]
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class VideoRecord(BaseModelExcludeNull):
|
||||||
|
"""Deserialized video record without metadata."""
|
||||||
|
|
||||||
|
video_id: str = Field(description="The unique id of the video.")
|
||||||
|
"""The unique id of the video."""
|
||||||
|
width: int = Field(description="The width of the video in px.")
|
||||||
|
"""The actual width of the video in px. This may be different from the width in metadata."""
|
||||||
|
height: int = Field(description="The height of the video in px.")
|
||||||
|
"""The actual height of the video in px. This may be different from the height in metadata."""
|
||||||
|
created_at: Union[datetime.datetime, str] = Field(description="The created timestamp of the video.")
|
||||||
|
"""The created timestamp of the video."""
|
||||||
|
updated_at: Union[datetime.datetime, str] = Field(description="The updated timestamp of the video.")
|
||||||
|
"""The updated timestamp of the video."""
|
||||||
|
deleted_at: Optional[Union[datetime.datetime, str]] = Field(
|
||||||
|
default=None, description="The deleted timestamp of the video."
|
||||||
|
)
|
||||||
|
"""The deleted timestamp of the video."""
|
||||||
|
is_intermediate: bool = Field(description="Whether this is an intermediate video.")
|
||||||
|
"""Whether this is an intermediate video."""
|
||||||
|
session_id: Optional[str] = Field(
|
||||||
|
default=None,
|
||||||
|
description="The session ID that generated this video, if it is a generated video.",
|
||||||
|
)
|
||||||
|
"""The session ID that generated this video, if it is a generated video."""
|
||||||
|
node_id: Optional[str] = Field(
|
||||||
|
default=None,
|
||||||
|
description="The node ID that generated this video, if it is a generated video.",
|
||||||
|
)
|
||||||
|
"""The node ID that generated this video, if it is a generated video."""
|
||||||
|
starred: bool = Field(description="Whether this video is starred.")
|
||||||
|
"""Whether this video is starred."""
|
||||||
|
|
||||||
|
|
||||||
|
class VideoRecordChanges(BaseModelExcludeNull):
|
||||||
|
"""A set of changes to apply to a video record.
|
||||||
|
|
||||||
|
Only limited changes are valid:
|
||||||
|
- `session_id`: change the session associated with a video
|
||||||
|
- `is_intermediate`: change the video's `is_intermediate` flag
|
||||||
|
- `starred`: change whether the video is starred
|
||||||
|
"""
|
||||||
|
|
||||||
|
session_id: Optional[StrictStr] = Field(
|
||||||
|
default=None,
|
||||||
|
description="The video's new session ID.",
|
||||||
|
)
|
||||||
|
"""The video's new session ID."""
|
||||||
|
is_intermediate: Optional[StrictBool] = Field(default=None, description="The video's new `is_intermediate` flag.")
|
||||||
|
"""The video's new `is_intermediate` flag."""
|
||||||
|
starred: Optional[StrictBool] = Field(default=None, description="The video's new `starred` state")
|
||||||
|
"""The video's new `starred` state."""
|
||||||
|
|
||||||
|
|
||||||
|
def deserialize_video_record(video_dict: dict) -> VideoRecord:
|
||||||
|
"""Deserializes a video record."""
|
||||||
|
|
||||||
|
# Retrieve all the values, setting "reasonable" defaults if they are not present.
|
||||||
|
video_id = video_dict.get("video_id", "unknown")
|
||||||
|
width = video_dict.get("width", 0)
|
||||||
|
height = video_dict.get("height", 0)
|
||||||
|
session_id = video_dict.get("session_id", None)
|
||||||
|
node_id = video_dict.get("node_id", None)
|
||||||
|
created_at = video_dict.get("created_at", get_iso_timestamp())
|
||||||
|
updated_at = video_dict.get("updated_at", get_iso_timestamp())
|
||||||
|
deleted_at = video_dict.get("deleted_at", get_iso_timestamp())
|
||||||
|
is_intermediate = video_dict.get("is_intermediate", False)
|
||||||
|
starred = video_dict.get("starred", False)
|
||||||
|
|
||||||
|
return VideoRecord(
|
||||||
|
video_id=video_id,
|
||||||
|
width=width,
|
||||||
|
height=height,
|
||||||
|
session_id=session_id,
|
||||||
|
node_id=node_id,
|
||||||
|
created_at=created_at,
|
||||||
|
updated_at=updated_at,
|
||||||
|
deleted_at=deleted_at,
|
||||||
|
is_intermediate=is_intermediate,
|
||||||
|
starred=starred,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class VideoCollectionCounts(BaseModel):
|
||||||
|
starred_count: int = Field(description="The number of starred videos in the collection.")
|
||||||
|
unstarred_count: int = Field(description="The number of unstarred videos in the collection.")
|
||||||
|
|
||||||
|
|
||||||
|
class VideoIdsResult(BaseModel):
|
||||||
|
"""Response containing ordered video ids with metadata for optimistic updates."""
|
||||||
|
|
||||||
|
video_ids: list[str] = Field(description="Ordered list of video ids")
|
||||||
|
starred_count: int = Field(description="Number of starred videos (when starred_first=True)")
|
||||||
|
total_count: int = Field(description="Total number of videos matching the query")
|
||||||
|
|
||||||
|
|
||||||
|
class VideoUrlsDTO(BaseModelExcludeNull):
|
||||||
|
"""The URLs for an image and its thumbnail."""
|
||||||
|
|
||||||
|
video_id: str = Field(description="The unique id of the video.")
|
||||||
|
"""The unique id of the video."""
|
||||||
|
video_url: str = Field(description="The URL of the video.")
|
||||||
|
"""The URL of the video."""
|
||||||
|
thumbnail_url: str = Field(description="The URL of the video's thumbnail.")
|
||||||
|
"""The URL of the video's thumbnail."""
|
||||||
|
|
||||||
|
|
||||||
|
class VideoDTO(VideoRecord, VideoUrlsDTO):
|
||||||
|
"""Deserialized video record, enriched for the frontend."""
|
||||||
|
|
||||||
|
board_id: Optional[str] = Field(
|
||||||
|
default=None, description="The id of the board the image belongs to, if one exists."
|
||||||
|
)
|
||||||
|
"""The id of the board the image belongs to, if one exists."""
|
||||||
|
|
||||||
|
|
||||||
|
def video_record_to_dto(
|
||||||
|
video_record: VideoRecord,
|
||||||
|
video_url: str,
|
||||||
|
thumbnail_url: str,
|
||||||
|
board_id: Optional[str],
|
||||||
|
) -> VideoDTO:
|
||||||
|
"""Converts a video record to a video DTO."""
|
||||||
|
return VideoDTO(
|
||||||
|
**video_record.model_dump(),
|
||||||
|
video_url=video_url,
|
||||||
|
thumbnail_url=thumbnail_url,
|
||||||
|
board_id=board_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ResultWithAffectedBoards(BaseModel):
|
||||||
|
affected_boards: list[str] = Field(description="The ids of boards affected by the delete operation")
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteVideosResult(ResultWithAffectedBoards):
|
||||||
|
deleted_videos: list[str] = Field(description="The ids of the videos that were deleted")
|
||||||
|
|
||||||
|
|
||||||
|
class StarredVideosResult(ResultWithAffectedBoards):
|
||||||
|
starred_videos: list[str] = Field(description="The ids of the videos that were starred")
|
||||||
|
|
||||||
|
|
||||||
|
class UnstarredVideosResult(ResultWithAffectedBoards):
|
||||||
|
unstarred_videos: list[str] = Field(description="The ids of the videos that were unstarred")
|
||||||
|
|
||||||
|
|
||||||
|
class AddVideosToBoardResult(ResultWithAffectedBoards):
|
||||||
|
added_videos: list[str] = Field(description="The video ids that were added to the board")
|
||||||
|
|
||||||
|
|
||||||
|
class RemoveVideosFromBoardResult(ResultWithAffectedBoards):
|
||||||
|
removed_videos: list[str] = Field(description="The video ids that were removed from their board")
|
||||||
@@ -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
|
import torch
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
from transformers.models.sam import SamModel
|
from transformers.models.sam import SamModel
|
||||||
from transformers.models.sam.processing_sam import SamProcessor
|
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
|
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):
|
class SegmentAnythingPipeline(RawModel):
|
||||||
"""A wrapper class for the transformers SAM model and processor that makes it compatible with the model manager."""
|
"""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(
|
def segment(
|
||||||
self,
|
self,
|
||||||
image: Image.Image,
|
image: Image.Image,
|
||||||
bounding_boxes: list[list[int]] | None = None,
|
inputs: list[SAMInput],
|
||||||
point_lists: list[list[list[int]]] | None = None,
|
|
||||||
) -> torch.Tensor:
|
) -> torch.Tensor:
|
||||||
"""Run the SAM model.
|
"""Segment the image using the provided inputs.
|
||||||
|
|
||||||
Either bounding_boxes or point_lists must be provided. If both are provided, bounding_boxes will be used and
|
|
||||||
point_lists will be ignored.
|
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
image (Image.Image): The image to segment.
|
image: The image to segment.
|
||||||
bounding_boxes (list[list[int]]): The bounding box prompts. Each bounding box is in the format
|
inputs: A list of SAMInput objects containing bounding boxes and/or point lists.
|
||||||
[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.
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
torch.Tensor: The segmentation masks. dtype: torch.bool. shape: [num_masks, channels, height, width].
|
torch.Tensor: The segmentation masks. dtype: torch.bool. shape: [num_masks, channels, height, width].
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Prep the inputs:
|
input_boxes: list[list[float]] = []
|
||||||
# - Create a list of bounding boxes or points and labels.
|
input_points: list[list[list[float]]] = []
|
||||||
# - Add a batch dimension of 1 to the inputs.
|
input_labels: list[list[int]] = []
|
||||||
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])
|
|
||||||
|
|
||||||
else:
|
for i in inputs:
|
||||||
raise ValueError("Either bounding_boxes or points and labels must be provided.")
|
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,
|
images=image,
|
||||||
input_boxes=input_boxes,
|
input_boxes=batched_input_boxes,
|
||||||
input_points=input_points,
|
input_points=batched_input_points,
|
||||||
input_labels=input_labels,
|
input_labels=batched_input_labels,
|
||||||
return_tensors="pt",
|
return_tensors="pt",
|
||||||
).to(self._sam_model.device)
|
).to(self._sam_model.device)
|
||||||
outputs = self._sam_model(**inputs)
|
outputs = self._sam_model(**processed_inputs)
|
||||||
masks = self._sam_processor.post_process_masks(
|
masks = self._sam_processor.post_process_masks(
|
||||||
masks=outputs.pred_masks,
|
masks=outputs.pred_masks,
|
||||||
original_sizes=inputs.original_sizes,
|
original_sizes=processed_inputs.original_sizes,
|
||||||
reshaped_input_sizes=inputs.reshaped_input_sizes,
|
reshaped_input_sizes=processed_inputs.reshaped_input_sizes,
|
||||||
)
|
)
|
||||||
|
|
||||||
# There should be only one batch.
|
# 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
|
||||||
@@ -90,6 +90,11 @@ class MainModelDefaultSettings(BaseModel):
|
|||||||
model_config = ConfigDict(extra="forbid")
|
model_config = ConfigDict(extra="forbid")
|
||||||
|
|
||||||
|
|
||||||
|
class LoraModelDefaultSettings(BaseModel):
|
||||||
|
weight: float | None = Field(default=None, ge=-1, le=2, description="Default weight for this model")
|
||||||
|
model_config = ConfigDict(extra="forbid")
|
||||||
|
|
||||||
|
|
||||||
class ControlAdapterDefaultSettings(BaseModel):
|
class ControlAdapterDefaultSettings(BaseModel):
|
||||||
# This could be narrowed to controlnet processor nodes, but they change. Leaving this a string is safer.
|
# This could be narrowed to controlnet processor nodes, but they change. Leaving this a string is safer.
|
||||||
preprocessor: str | None
|
preprocessor: str | None
|
||||||
@@ -287,6 +292,9 @@ class LoRAConfigBase(ABC, BaseModel):
|
|||||||
|
|
||||||
type: Literal[ModelType.LoRA] = ModelType.LoRA
|
type: Literal[ModelType.LoRA] = ModelType.LoRA
|
||||||
trigger_phrases: Optional[set[str]] = Field(description="Set of trigger phrases for this model", default=None)
|
trigger_phrases: Optional[set[str]] = Field(description="Set of trigger phrases for this model", default=None)
|
||||||
|
default_settings: Optional[LoraModelDefaultSettings] = Field(
|
||||||
|
description="Default settings for this model", default=None
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def flux_lora_format(cls, mod: ModelOnDisk):
|
def flux_lora_format(cls, mod: ModelOnDisk):
|
||||||
@@ -492,6 +500,15 @@ class MainConfigBase(ABC, BaseModel):
|
|||||||
variant: AnyVariant = ModelVariantType.Normal
|
variant: AnyVariant = ModelVariantType.Normal
|
||||||
|
|
||||||
|
|
||||||
|
class VideoConfigBase(ABC, BaseModel):
|
||||||
|
type: Literal[ModelType.Video] = ModelType.Video
|
||||||
|
trigger_phrases: Optional[set[str]] = Field(description="Set of trigger phrases for this model", default=None)
|
||||||
|
default_settings: Optional[MainModelDefaultSettings] = Field(
|
||||||
|
description="Default settings for this model", default=None
|
||||||
|
)
|
||||||
|
variant: AnyVariant = ModelVariantType.Normal
|
||||||
|
|
||||||
|
|
||||||
class MainCheckpointConfig(CheckpointConfigBase, MainConfigBase, LegacyProbeMixin, ModelConfigBase):
|
class MainCheckpointConfig(CheckpointConfigBase, MainConfigBase, LegacyProbeMixin, ModelConfigBase):
|
||||||
"""Model config for main checkpoint models."""
|
"""Model config for main checkpoint models."""
|
||||||
|
|
||||||
@@ -649,6 +666,21 @@ class ApiModelConfig(MainConfigBase, ModelConfigBase):
|
|||||||
raise NotImplementedError("API models are not parsed from disk.")
|
raise NotImplementedError("API models are not parsed from disk.")
|
||||||
|
|
||||||
|
|
||||||
|
class VideoApiModelConfig(VideoConfigBase, ModelConfigBase):
|
||||||
|
"""Model config for API-based video models."""
|
||||||
|
|
||||||
|
format: Literal[ModelFormat.Api] = ModelFormat.Api
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def matches(cls, mod: ModelOnDisk) -> bool:
|
||||||
|
# API models are not stored on disk, so we can't match them.
|
||||||
|
return False
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def parse(cls, mod: ModelOnDisk) -> dict[str, Any]:
|
||||||
|
raise NotImplementedError("API models are not parsed from disk.")
|
||||||
|
|
||||||
|
|
||||||
def get_model_discriminator_value(v: Any) -> str:
|
def get_model_discriminator_value(v: Any) -> str:
|
||||||
"""
|
"""
|
||||||
Computes the discriminator value for a model config.
|
Computes the discriminator value for a model config.
|
||||||
@@ -718,12 +750,13 @@ AnyModelConfig = Annotated[
|
|||||||
Annotated[FluxReduxConfig, FluxReduxConfig.get_tag()],
|
Annotated[FluxReduxConfig, FluxReduxConfig.get_tag()],
|
||||||
Annotated[LlavaOnevisionConfig, LlavaOnevisionConfig.get_tag()],
|
Annotated[LlavaOnevisionConfig, LlavaOnevisionConfig.get_tag()],
|
||||||
Annotated[ApiModelConfig, ApiModelConfig.get_tag()],
|
Annotated[ApiModelConfig, ApiModelConfig.get_tag()],
|
||||||
|
Annotated[VideoApiModelConfig, VideoApiModelConfig.get_tag()],
|
||||||
],
|
],
|
||||||
Discriminator(get_model_discriminator_value),
|
Discriminator(get_model_discriminator_value),
|
||||||
]
|
]
|
||||||
|
|
||||||
AnyModelConfigValidator = TypeAdapter(AnyModelConfig)
|
AnyModelConfigValidator = TypeAdapter(AnyModelConfig)
|
||||||
AnyDefaultSettings: TypeAlias = Union[MainModelDefaultSettings, ControlAdapterDefaultSettings]
|
AnyDefaultSettings: TypeAlias = Union[MainModelDefaultSettings, LoraModelDefaultSettings, ControlAdapterDefaultSettings]
|
||||||
|
|
||||||
|
|
||||||
class ModelConfigFactory:
|
class ModelConfigFactory:
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ from invokeai.backend.model_manager.config import (
|
|||||||
AnyModelConfig,
|
AnyModelConfig,
|
||||||
ControlAdapterDefaultSettings,
|
ControlAdapterDefaultSettings,
|
||||||
InvalidModelConfigException,
|
InvalidModelConfigException,
|
||||||
|
LoraModelDefaultSettings,
|
||||||
MainModelDefaultSettings,
|
MainModelDefaultSettings,
|
||||||
ModelConfigFactory,
|
ModelConfigFactory,
|
||||||
SubmodelDefinition,
|
SubmodelDefinition,
|
||||||
@@ -217,6 +218,8 @@ class ModelProbe(object):
|
|||||||
if not fields["default_settings"]:
|
if not fields["default_settings"]:
|
||||||
if fields["type"] in {ModelType.ControlNet, ModelType.T2IAdapter, ModelType.ControlLoRa}:
|
if fields["type"] in {ModelType.ControlNet, ModelType.T2IAdapter, ModelType.ControlLoRa}:
|
||||||
fields["default_settings"] = get_default_settings_control_adapters(fields["name"])
|
fields["default_settings"] = get_default_settings_control_adapters(fields["name"])
|
||||||
|
if fields["type"] in {ModelType.LoRA}:
|
||||||
|
fields["default_settings"] = get_default_settings_lora()
|
||||||
elif fields["type"] is ModelType.Main:
|
elif fields["type"] is ModelType.Main:
|
||||||
fields["default_settings"] = get_default_settings_main(fields["base"])
|
fields["default_settings"] = get_default_settings_main(fields["base"])
|
||||||
|
|
||||||
@@ -543,6 +546,10 @@ def get_default_settings_control_adapters(model_name: str) -> Optional[ControlAd
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def get_default_settings_lora() -> LoraModelDefaultSettings:
|
||||||
|
return LoraModelDefaultSettings()
|
||||||
|
|
||||||
|
|
||||||
def get_default_settings_main(model_base: BaseModelType) -> Optional[MainModelDefaultSettings]:
|
def get_default_settings_main(model_base: BaseModelType) -> Optional[MainModelDefaultSettings]:
|
||||||
if model_base is BaseModelType.StableDiffusion1 or model_base is BaseModelType.StableDiffusion2:
|
if model_base is BaseModelType.StableDiffusion1 or model_base is BaseModelType.StableDiffusion2:
|
||||||
return MainModelDefaultSettings(width=512, height=512)
|
return MainModelDefaultSettings(width=512, height=512)
|
||||||
|
|||||||
@@ -31,6 +31,8 @@ class BaseModelType(str, Enum):
|
|||||||
Gemini2_5 = "gemini-2.5"
|
Gemini2_5 = "gemini-2.5"
|
||||||
ChatGPT4o = "chatgpt-4o"
|
ChatGPT4o = "chatgpt-4o"
|
||||||
FluxKontext = "flux-kontext"
|
FluxKontext = "flux-kontext"
|
||||||
|
Veo3 = "veo3"
|
||||||
|
Runway = "runway"
|
||||||
|
|
||||||
|
|
||||||
class ModelType(str, Enum):
|
class ModelType(str, Enum):
|
||||||
@@ -52,6 +54,7 @@ class ModelType(str, Enum):
|
|||||||
SigLIP = "siglip"
|
SigLIP = "siglip"
|
||||||
FluxRedux = "flux_redux"
|
FluxRedux = "flux_redux"
|
||||||
LlavaOnevision = "llava_onevision"
|
LlavaOnevision = "llava_onevision"
|
||||||
|
Video = "video"
|
||||||
|
|
||||||
|
|
||||||
class SubModelType(str, Enum):
|
class SubModelType(str, Enum):
|
||||||
|
|||||||
@@ -5,7 +5,11 @@ import torch
|
|||||||
from diffusers.configuration_utils import ConfigMixin, register_to_config
|
from diffusers.configuration_utils import ConfigMixin, register_to_config
|
||||||
from diffusers.loaders.single_file_model import FromOriginalModelMixin
|
from diffusers.loaders.single_file_model import FromOriginalModelMixin
|
||||||
from diffusers.models.attention_processor import AttentionProcessor, AttnProcessor
|
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 (
|
from diffusers.models.embeddings import (
|
||||||
TextImageProjection,
|
TextImageProjection,
|
||||||
TextImageTimeEmbedding,
|
TextImageTimeEmbedding,
|
||||||
@@ -775,7 +779,15 @@ class ControlNetModel(ModelMixin, ConfigMixin, FromOriginalModelMixin):
|
|||||||
|
|
||||||
|
|
||||||
diffusers.ControlNetModel = ControlNetModel
|
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
|
# 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/dagre": "^1.1.5",
|
||||||
"@dagrejs/graphlib": "^2.2.4",
|
"@dagrejs/graphlib": "^2.2.4",
|
||||||
"@fontsource-variable/inter": "^5.2.6",
|
"@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",
|
"@nanostores/react": "^1.0.0",
|
||||||
"@observ33r/object-equals": "^1.1.5",
|
"@observ33r/object-equals": "^1.1.5",
|
||||||
"@reduxjs/toolkit": "2.8.2",
|
"@reduxjs/toolkit": "2.8.2",
|
||||||
@@ -56,7 +56,7 @@
|
|||||||
"chakra-react-select": "^4.9.2",
|
"chakra-react-select": "^4.9.2",
|
||||||
"cmdk": "^1.1.1",
|
"cmdk": "^1.1.1",
|
||||||
"compare-versions": "^6.1.1",
|
"compare-versions": "^6.1.1",
|
||||||
"dockview": "^4.4.1",
|
"dockview": "^4.7.1",
|
||||||
"es-toolkit": "^1.39.7",
|
"es-toolkit": "^1.39.7",
|
||||||
"filesize": "^10.1.6",
|
"filesize": "^10.1.6",
|
||||||
"fracturedjsonjs": "^4.1.0",
|
"fracturedjsonjs": "^4.1.0",
|
||||||
@@ -69,6 +69,7 @@
|
|||||||
"linkify-react": "^4.3.1",
|
"linkify-react": "^4.3.1",
|
||||||
"linkifyjs": "^4.3.1",
|
"linkifyjs": "^4.3.1",
|
||||||
"lru-cache": "^11.1.0",
|
"lru-cache": "^11.1.0",
|
||||||
|
"media-chrome": "^4.13.0",
|
||||||
"mtwist": "^1.0.2",
|
"mtwist": "^1.0.2",
|
||||||
"nanoid": "^5.1.5",
|
"nanoid": "^5.1.5",
|
||||||
"nanostores": "^1.0.1",
|
"nanostores": "^1.0.1",
|
||||||
@@ -87,6 +88,7 @@
|
|||||||
"react-hotkeys-hook": "4.5.0",
|
"react-hotkeys-hook": "4.5.0",
|
||||||
"react-i18next": "^15.5.3",
|
"react-i18next": "^15.5.3",
|
||||||
"react-icons": "^5.5.0",
|
"react-icons": "^5.5.0",
|
||||||
|
"react-player": "^3.3.1",
|
||||||
"react-redux": "9.2.0",
|
"react-redux": "9.2.0",
|
||||||
"react-resizable-panels": "^3.0.3",
|
"react-resizable-panels": "^3.0.3",
|
||||||
"react-textarea-autosize": "^8.5.9",
|
"react-textarea-autosize": "^8.5.9",
|
||||||
|
|||||||
487
invokeai/frontend/web/pnpm-lock.yaml
generated
487
invokeai/frontend/web/pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -14,8 +14,7 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "حجم الصورة",
|
"galleryImageSize": "حجم الصورة",
|
||||||
"gallerySettings": "إعدادات المعرض",
|
"gallerySettings": "إعدادات المعرض",
|
||||||
"autoSwitchNewImages": "التبديل التلقائي إلى الصور الجديدة",
|
"autoSwitchNewImages": "التبديل التلقائي إلى الصور الجديدة"
|
||||||
"noImagesInGallery": "لا توجد صور في المعرض"
|
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
"modelManager": "مدير النموذج",
|
"modelManager": "مدير النموذج",
|
||||||
@@ -62,12 +61,10 @@
|
|||||||
"infillMethod": "طريقة التعبئة",
|
"infillMethod": "طريقة التعبئة",
|
||||||
"tileSize": "حجم البلاطة",
|
"tileSize": "حجم البلاطة",
|
||||||
"copyImage": "نسخ الصورة",
|
"copyImage": "نسخ الصورة",
|
||||||
"downloadImage": "تحميل الصورة",
|
|
||||||
"usePrompt": "استخدم المحث",
|
"usePrompt": "استخدم المحث",
|
||||||
"useSeed": "استخدام البذور",
|
"useSeed": "استخدام البذور",
|
||||||
"useAll": "استخدام الكل",
|
"useAll": "استخدام الكل",
|
||||||
"info": "معلومات",
|
"info": "معلومات"
|
||||||
"showOptionsPanel": "إظهار لوحة الخيارات"
|
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"models": "موديلات",
|
"models": "موديلات",
|
||||||
|
|||||||
@@ -24,7 +24,6 @@
|
|||||||
"ipAdapter": "IP Adapter",
|
"ipAdapter": "IP Adapter",
|
||||||
"auto": "Auto",
|
"auto": "Auto",
|
||||||
"controlNet": "ControlNet",
|
"controlNet": "ControlNet",
|
||||||
"imageFailedToLoad": "Kann Bild nicht laden",
|
|
||||||
"modelManager": "Model Manager",
|
"modelManager": "Model Manager",
|
||||||
"learnMore": "Mehr erfahren",
|
"learnMore": "Mehr erfahren",
|
||||||
"loading": "Lade",
|
"loading": "Lade",
|
||||||
@@ -52,7 +51,6 @@
|
|||||||
"somethingWentWrong": "Etwas ist schief gelaufen",
|
"somethingWentWrong": "Etwas ist schief gelaufen",
|
||||||
"copyError": "$t(gallery.copy) Fehler",
|
"copyError": "$t(gallery.copy) Fehler",
|
||||||
"input": "Eingabe",
|
"input": "Eingabe",
|
||||||
"notInstalled": "Nicht $t(common.installed)",
|
|
||||||
"alpha": "Alpha",
|
"alpha": "Alpha",
|
||||||
"red": "Rot",
|
"red": "Rot",
|
||||||
"green": "Grün",
|
"green": "Grün",
|
||||||
@@ -62,11 +60,8 @@
|
|||||||
"direction": "Richtung",
|
"direction": "Richtung",
|
||||||
"save": "Speichern",
|
"save": "Speichern",
|
||||||
"created": "Erstellt",
|
"created": "Erstellt",
|
||||||
"prevPage": "Vorherige Seite",
|
|
||||||
"nextPage": "Nächste Seite",
|
|
||||||
"unknownError": "Unbekannter Fehler",
|
"unknownError": "Unbekannter Fehler",
|
||||||
"aboutDesc": "Verwenden Sie Invoke für die Arbeit? Siehe hier:",
|
"aboutDesc": "Verwenden Sie Invoke für die Arbeit? Siehe hier:",
|
||||||
"localSystem": "Lokales System",
|
|
||||||
"orderBy": "Ordnen nach",
|
"orderBy": "Ordnen nach",
|
||||||
"saveAs": "Speichern als",
|
"saveAs": "Speichern als",
|
||||||
"updated": "Aktualisiert",
|
"updated": "Aktualisiert",
|
||||||
@@ -77,7 +72,6 @@
|
|||||||
"selected": "Ausgewählt",
|
"selected": "Ausgewählt",
|
||||||
"beta": "Beta",
|
"beta": "Beta",
|
||||||
"editor": "Editor",
|
"editor": "Editor",
|
||||||
"goTo": "Gehe zu",
|
|
||||||
"positivePrompt": "Positiv-Prompt",
|
"positivePrompt": "Positiv-Prompt",
|
||||||
"negativePrompt": "Negativ-Prompt",
|
"negativePrompt": "Negativ-Prompt",
|
||||||
"tab": "Tabulator",
|
"tab": "Tabulator",
|
||||||
@@ -106,7 +100,6 @@
|
|||||||
"values": "Werte",
|
"values": "Werte",
|
||||||
"min": "Min",
|
"min": "Min",
|
||||||
"max": "Max",
|
"max": "Max",
|
||||||
"resetToDefaults": "Auf Standard zurücksetzen",
|
|
||||||
"seed": "Seed",
|
"seed": "Seed",
|
||||||
"row": "Reihe",
|
"row": "Reihe",
|
||||||
"column": "Spalte",
|
"column": "Spalte",
|
||||||
@@ -135,14 +128,12 @@
|
|||||||
"galleryImageSize": "Bildgröße",
|
"galleryImageSize": "Bildgröße",
|
||||||
"gallerySettings": "Galerie-Einstellungen",
|
"gallerySettings": "Galerie-Einstellungen",
|
||||||
"autoSwitchNewImages": "Auto-Wechsel zu neuen Bildern",
|
"autoSwitchNewImages": "Auto-Wechsel zu neuen Bildern",
|
||||||
"noImagesInGallery": "Keine Bilder in der Galerie",
|
|
||||||
"loading": "Lade",
|
"loading": "Lade",
|
||||||
"deleteImage_one": "Lösche Bild",
|
"deleteImage_one": "Lösche Bild",
|
||||||
"deleteImage_other": "Lösche {{count}} Bilder",
|
"deleteImage_other": "Lösche {{count}} Bilder",
|
||||||
"copy": "Kopieren",
|
"copy": "Kopieren",
|
||||||
"download": "Runterladen",
|
"download": "Runterladen",
|
||||||
"featuresWillReset": "Wenn Sie dieses Bild löschen, werden diese Funktionen sofort zurückgesetzt.",
|
"featuresWillReset": "Wenn Sie dieses Bild löschen, werden diese Funktionen sofort zurückgesetzt.",
|
||||||
"unableToLoad": "Galerie kann nicht geladen werden",
|
|
||||||
"downloadSelection": "Auswahl herunterladen",
|
"downloadSelection": "Auswahl herunterladen",
|
||||||
"currentlyInUse": "Dieses Bild wird derzeit in den folgenden Funktionen verwendet:",
|
"currentlyInUse": "Dieses Bild wird derzeit in den folgenden Funktionen verwendet:",
|
||||||
"deleteImagePermanent": "Gelöschte Bilder können nicht wiederhergestellt werden.",
|
"deleteImagePermanent": "Gelöschte Bilder können nicht wiederhergestellt werden.",
|
||||||
@@ -182,16 +173,12 @@
|
|||||||
"gallery": "Galerie",
|
"gallery": "Galerie",
|
||||||
"sortDirection": "Sortierreihenfolge",
|
"sortDirection": "Sortierreihenfolge",
|
||||||
"sideBySide": "Nebeneinander",
|
"sideBySide": "Nebeneinander",
|
||||||
"openViewer": "Viewer öffnen",
|
|
||||||
"viewerImage": "Viewer-Bild",
|
"viewerImage": "Viewer-Bild",
|
||||||
"exitCompare": "Vergleichen beenden",
|
"exitCompare": "Vergleichen beenden",
|
||||||
"closeViewer": "Viewer schließen",
|
|
||||||
"selectAnImageToCompare": "Wählen Sie ein Bild zum Vergleichen",
|
|
||||||
"stretchToFit": "Strecken bis es passt",
|
"stretchToFit": "Strecken bis es passt",
|
||||||
"displayBoardSearch": "Board durchsuchen",
|
"displayBoardSearch": "Board durchsuchen",
|
||||||
"displaySearch": "Bild suchen",
|
"displaySearch": "Bild suchen",
|
||||||
"go": "Los",
|
"go": "Los",
|
||||||
"jump": "Springen",
|
|
||||||
"assetsTab": "Dateien, die Sie zur Verwendung in Ihren Projekten hochgeladen haben.",
|
"assetsTab": "Dateien, die Sie zur Verwendung in Ihren Projekten hochgeladen haben.",
|
||||||
"imagesTab": "Bilder, die Sie in Invoke erstellt und gespeichert haben.",
|
"imagesTab": "Bilder, die Sie in Invoke erstellt und gespeichert haben.",
|
||||||
"boardsSettings": "Ordnereinstellungen",
|
"boardsSettings": "Ordnereinstellungen",
|
||||||
@@ -210,10 +197,6 @@
|
|||||||
"title": "Bbox Werkzeug",
|
"title": "Bbox Werkzeug",
|
||||||
"desc": "Bbox Werkzeug auswählen."
|
"desc": "Bbox Werkzeug auswählen."
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "Farbe auf Weiß einstellen",
|
|
||||||
"desc": "Setzt die aktuelle Werkzeugfarbe auf weiß."
|
|
||||||
},
|
|
||||||
"title": "Leinwand",
|
"title": "Leinwand",
|
||||||
"selectBrushTool": {
|
"selectBrushTool": {
|
||||||
"title": "Pinselwerkzeug",
|
"title": "Pinselwerkzeug",
|
||||||
@@ -578,7 +561,6 @@
|
|||||||
"urlOrLocalPath": "URL oder lokaler Pfad",
|
"urlOrLocalPath": "URL oder lokaler Pfad",
|
||||||
"install": "Installieren",
|
"install": "Installieren",
|
||||||
"textualInversions": "Textuelle Inversionen",
|
"textualInversions": "Textuelle Inversionen",
|
||||||
"ipAdapters": "IP-Adapter",
|
|
||||||
"modelImageUpdated": "Modellbild aktualisiert",
|
"modelImageUpdated": "Modellbild aktualisiert",
|
||||||
"path": "Pfad",
|
"path": "Pfad",
|
||||||
"pathToConfig": "Pfad zur Konfiguration",
|
"pathToConfig": "Pfad zur Konfiguration",
|
||||||
@@ -601,7 +583,6 @@
|
|||||||
"repoVariant": "Repo Variante",
|
"repoVariant": "Repo Variante",
|
||||||
"learnMoreAboutSupportedModels": "Erfahren Sie mehr über die Modelle, die wir unterstützen",
|
"learnMoreAboutSupportedModels": "Erfahren Sie mehr über die Modelle, die wir unterstützen",
|
||||||
"clipEmbed": "CLIP einbetten",
|
"clipEmbed": "CLIP einbetten",
|
||||||
"starterModelsInModelManager": "Modelle für Ihren Start finden Sie im Modell-Manager",
|
|
||||||
"noModelsInstalledDesc1": "Installiere Modelle mit dem",
|
"noModelsInstalledDesc1": "Installiere Modelle mit dem",
|
||||||
"modelImageUpdateFailed": "Modellbild-Update fehlgeschlagen",
|
"modelImageUpdateFailed": "Modellbild-Update fehlgeschlagen",
|
||||||
"prune": "Bereinigen",
|
"prune": "Bereinigen",
|
||||||
@@ -661,11 +642,9 @@
|
|||||||
"scaledHeight": "Skaliert H",
|
"scaledHeight": "Skaliert H",
|
||||||
"infillMethod": "Infill-Methode",
|
"infillMethod": "Infill-Methode",
|
||||||
"tileSize": "Kachelgröße",
|
"tileSize": "Kachelgröße",
|
||||||
"downloadImage": "Bild herunterladen",
|
|
||||||
"usePrompt": "Prompt verwenden",
|
"usePrompt": "Prompt verwenden",
|
||||||
"useSeed": "Seed verwenden",
|
"useSeed": "Seed verwenden",
|
||||||
"useAll": "Alle verwenden",
|
"useAll": "Alle verwenden",
|
||||||
"showOptionsPanel": "Optionsleiste zeigen",
|
|
||||||
"copyImage": "Bild kopieren",
|
"copyImage": "Bild kopieren",
|
||||||
"denoisingStrength": "Stärke der Entrauschung",
|
"denoisingStrength": "Stärke der Entrauschung",
|
||||||
"symmetry": "Symmetrie",
|
"symmetry": "Symmetrie",
|
||||||
@@ -681,10 +660,6 @@
|
|||||||
"remixImage": "Remix des Bilds erstellen",
|
"remixImage": "Remix des Bilds erstellen",
|
||||||
"imageActions": "Weitere Bildaktionen",
|
"imageActions": "Weitere Bildaktionen",
|
||||||
"invoke": {
|
"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",
|
"noNodesInGraph": "Keine Knoten im Graphen",
|
||||||
"canvasIsTransforming": "Leinwand ist beschäftigt (wird transformiert)",
|
"canvasIsTransforming": "Leinwand ist beschäftigt (wird transformiert)",
|
||||||
"canvasIsRasterizing": "Leinwand ist beschäftigt (wird gerastert)",
|
"canvasIsRasterizing": "Leinwand ist beschäftigt (wird gerastert)",
|
||||||
@@ -750,7 +725,6 @@
|
|||||||
"parametersNotSet": "Parameter nicht zurückgerufen",
|
"parametersNotSet": "Parameter nicht zurückgerufen",
|
||||||
"addedToBoard": "Dem Board hinzugefügt",
|
"addedToBoard": "Dem Board hinzugefügt",
|
||||||
"loadedWithWarnings": "Workflow mit Warnungen geladen",
|
"loadedWithWarnings": "Workflow mit Warnungen geladen",
|
||||||
"imageSaved": "Bild gespeichert",
|
|
||||||
"linkCopied": "Link kopiert",
|
"linkCopied": "Link kopiert",
|
||||||
"problemCopyingLayer": "Ebene kann nicht kopiert werden",
|
"problemCopyingLayer": "Ebene kann nicht kopiert werden",
|
||||||
"problemSavingLayer": "Ebene kann nicht gespeichert werden",
|
"problemSavingLayer": "Ebene kann nicht gespeichert werden",
|
||||||
@@ -761,8 +735,6 @@
|
|||||||
"prunedQueue": "Warteschlange bereinigt",
|
"prunedQueue": "Warteschlange bereinigt",
|
||||||
"modelAddedSimple": "Modell zur Warteschlange hinzugefügt",
|
"modelAddedSimple": "Modell zur Warteschlange hinzugefügt",
|
||||||
"parametersSet": "Parameter zurückgerufen",
|
"parametersSet": "Parameter zurückgerufen",
|
||||||
"imageNotLoadedDesc": "Bild konnte nicht gefunden werden",
|
|
||||||
"setControlImage": "Als Kontrollbild festlegen",
|
|
||||||
"sentToUpscale": "An Vergrößerung gesendet",
|
"sentToUpscale": "An Vergrößerung gesendet",
|
||||||
"parameterNotSetDescWithMessage": "{{parameter}} kann nicht zurückgerufen werden: {{message}}",
|
"parameterNotSetDescWithMessage": "{{parameter}} kann nicht zurückgerufen werden: {{message}}",
|
||||||
"unableToLoadImageMetadata": "Bildmetadaten können nicht geladen werden",
|
"unableToLoadImageMetadata": "Bildmetadaten können nicht geladen werden",
|
||||||
@@ -775,7 +747,6 @@
|
|||||||
"parameterSet": "Parameter zurückgerufen",
|
"parameterSet": "Parameter zurückgerufen",
|
||||||
"importFailed": "Import fehlgeschlagen",
|
"importFailed": "Import fehlgeschlagen",
|
||||||
"importSuccessful": "Import erfolgreich",
|
"importSuccessful": "Import erfolgreich",
|
||||||
"setNodeField": "Als Knotenfeld festlegen",
|
|
||||||
"somethingWentWrong": "Etwas ist schief gelaufen",
|
"somethingWentWrong": "Etwas ist schief gelaufen",
|
||||||
"workflowLoaded": "Arbeitsablauf geladen",
|
"workflowLoaded": "Arbeitsablauf geladen",
|
||||||
"workflowDeleted": "Arbeitsablauf gelöscht",
|
"workflowDeleted": "Arbeitsablauf gelöscht",
|
||||||
@@ -783,16 +754,12 @@
|
|||||||
"layerCopiedToClipboard": "Ebene in die Zwischenablage kopiert",
|
"layerCopiedToClipboard": "Ebene in die Zwischenablage kopiert",
|
||||||
"sentToCanvas": "An Leinwand gesendet",
|
"sentToCanvas": "An Leinwand gesendet",
|
||||||
"problemDeletingWorkflow": "Problem beim Löschen des Arbeitsablaufs",
|
"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",
|
"problemRetrievingWorkflow": "Problem beim Abrufen des Arbeitsablaufs",
|
||||||
"uploadFailedInvalidUploadDesc": "Müssen PNG-, JPEG- oder WEBP-Bilder sein.",
|
"uploadFailedInvalidUploadDesc": "Müssen PNG-, JPEG- oder WEBP-Bilder sein.",
|
||||||
"pasteSuccess": "Eingefügt in {{destination}}",
|
"pasteSuccess": "Eingefügt in {{destination}}",
|
||||||
"pasteFailed": "Einfügen fehlgeschlagen",
|
"pasteFailed": "Einfügen fehlgeschlagen",
|
||||||
"unableToCopy": "Kopieren nicht möglich",
|
"unableToCopy": "Kopieren nicht möglich",
|
||||||
"unableToCopyDesc_theseSteps": "diese Schritte",
|
"unableToCopyDesc_theseSteps": "diese Schritte",
|
||||||
"noRasterLayers": "Keine Rasterebenen gefunden",
|
|
||||||
"noActiveRasterLayers": "Keine aktiven Rasterebenen",
|
|
||||||
"noVisibleRasterLayers": "Keine sichtbaren Rasterebenen"
|
"noVisibleRasterLayers": "Keine sichtbaren Rasterebenen"
|
||||||
},
|
},
|
||||||
"accessibility": {
|
"accessibility": {
|
||||||
@@ -845,16 +812,13 @@
|
|||||||
"archiveBoard": "Ordner archivieren",
|
"archiveBoard": "Ordner archivieren",
|
||||||
"archived": "Archiviert",
|
"archived": "Archiviert",
|
||||||
"noBoards": "Kein {{boardType}} Ordner",
|
"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.",
|
"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_one": "{{count}} in der Sammlung",
|
||||||
"assetsWithCount_other": "{{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.",
|
"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",
|
"updateBoardError": "Fehler beim Aktualisieren des Ordners",
|
||||||
"uncategorizedImages": "Nicht kategorisierte Bilder",
|
"uncategorizedImages": "Nicht kategorisierte Bilder",
|
||||||
"deleteAllUncategorizedImages": "Alle nicht kategorisierten Bilder löschen",
|
"deleteAllUncategorizedImages": "Alle nicht kategorisierten Bilder löschen"
|
||||||
"deletedImagesCannotBeRestored": "Gelöschte Bilder können nicht wiederhergestellt werden."
|
|
||||||
},
|
},
|
||||||
"queue": {
|
"queue": {
|
||||||
"status": "Status",
|
"status": "Status",
|
||||||
@@ -909,7 +873,6 @@
|
|||||||
"batchQueuedDesc_other": "{{count}} Einträge an {{direction}} der Wartschlange hinzugefügt",
|
"batchQueuedDesc_other": "{{count}} Einträge an {{direction}} der Wartschlange hinzugefügt",
|
||||||
"openQueue": "Warteschlange öffnen",
|
"openQueue": "Warteschlange öffnen",
|
||||||
"batchFailedToQueue": "Fehler beim Einreihen in die Stapelverarbeitung",
|
"batchFailedToQueue": "Fehler beim Einreihen in die Stapelverarbeitung",
|
||||||
"batchFieldValues": "Stapelverarbeitungswerte",
|
|
||||||
"batchQueued": "Stapelverarbeitung eingereiht",
|
"batchQueued": "Stapelverarbeitung eingereiht",
|
||||||
"graphQueued": "Graph eingereiht",
|
"graphQueued": "Graph eingereiht",
|
||||||
"graphFailedToQueue": "Fehler beim Einreihen des Graphen",
|
"graphFailedToQueue": "Fehler beim Einreihen des Graphen",
|
||||||
@@ -956,8 +919,6 @@
|
|||||||
"allPrompts": "Alle Prompts",
|
"allPrompts": "Alle Prompts",
|
||||||
"imageDimensions": "Bilder Auslösungen",
|
"imageDimensions": "Bilder Auslösungen",
|
||||||
"parameterSet": "Parameter {{parameter}} setzen",
|
"parameterSet": "Parameter {{parameter}} setzen",
|
||||||
"recallParameter": "{{label}} Abrufen",
|
|
||||||
"parsingFailed": "Parsing Fehlgeschlagen",
|
|
||||||
"canvasV2Metadata": "Leinwand",
|
"canvasV2Metadata": "Leinwand",
|
||||||
"guidance": "Führung",
|
"guidance": "Führung",
|
||||||
"seamlessXAxis": "Nahtlose X Achse",
|
"seamlessXAxis": "Nahtlose X Achse",
|
||||||
@@ -1240,9 +1201,7 @@
|
|||||||
"collectionFieldType": "{{name}} (Sammlung)",
|
"collectionFieldType": "{{name}} (Sammlung)",
|
||||||
"connectionWouldCreateCycle": "Verbindung würde einen Kreislauf/cycle schaffen",
|
"connectionWouldCreateCycle": "Verbindung würde einen Kreislauf/cycle schaffen",
|
||||||
"inputMayOnlyHaveOneConnection": "Eingang darf nur eine Verbindung haben",
|
"inputMayOnlyHaveOneConnection": "Eingang darf nur eine Verbindung haben",
|
||||||
"hideLegendNodes": "Feldtyp-Legende ausblenden",
|
|
||||||
"integer": "Ganze Zahl",
|
"integer": "Ganze Zahl",
|
||||||
"addLinearView": "Zur linearen Ansicht hinzufügen",
|
|
||||||
"currentImageDescription": "Zeigt das aktuelle Bild im Node-Editor an",
|
"currentImageDescription": "Zeigt das aktuelle Bild im Node-Editor an",
|
||||||
"ipAdapter": "IP-Adapter",
|
"ipAdapter": "IP-Adapter",
|
||||||
"hideMinimapnodes": "Miniatur-Kartenansicht ausblenden",
|
"hideMinimapnodes": "Miniatur-Kartenansicht ausblenden",
|
||||||
@@ -1251,7 +1210,6 @@
|
|||||||
"reloadNodeTemplates": "Knoten-Vorlagen neu laden",
|
"reloadNodeTemplates": "Knoten-Vorlagen neu laden",
|
||||||
"newWorkflow": "Neuer Arbeitsablauf / Workflow",
|
"newWorkflow": "Neuer Arbeitsablauf / Workflow",
|
||||||
"newWorkflowDesc": "Einen neuen Arbeitsablauf erstellen?",
|
"newWorkflowDesc": "Einen neuen Arbeitsablauf erstellen?",
|
||||||
"noFieldsLinearview": "Keine Felder zur linearen Ansicht hinzugefügt",
|
|
||||||
"clearWorkflow": "Workflow löschen",
|
"clearWorkflow": "Workflow löschen",
|
||||||
"clearWorkflowDesc": "Diesen Arbeitsablauf löschen und neu starten?",
|
"clearWorkflowDesc": "Diesen Arbeitsablauf löschen und neu starten?",
|
||||||
"noConnectionInProgress": "Es besteht keine Verbindung",
|
"noConnectionInProgress": "Es besteht keine Verbindung",
|
||||||
@@ -1259,7 +1217,6 @@
|
|||||||
"nodeVersion": "Knoten Version",
|
"nodeVersion": "Knoten Version",
|
||||||
"node": "Knoten",
|
"node": "Knoten",
|
||||||
"nodeSearch": "Knoten suchen",
|
"nodeSearch": "Knoten suchen",
|
||||||
"removeLinearView": "Entfernen aus Linear View",
|
|
||||||
"nodeOutputs": "Knoten-Ausgänge",
|
"nodeOutputs": "Knoten-Ausgänge",
|
||||||
"nodeTemplate": "Knoten-Vorlage",
|
"nodeTemplate": "Knoten-Vorlage",
|
||||||
"nodeType": "Knotentyp",
|
"nodeType": "Knotentyp",
|
||||||
@@ -1270,7 +1227,6 @@
|
|||||||
"clearWorkflowDesc2": "Ihr aktueller Arbeitsablauf hat ungespeicherte Änderungen.",
|
"clearWorkflowDesc2": "Ihr aktueller Arbeitsablauf hat ungespeicherte Änderungen.",
|
||||||
"scheduler": "Planer",
|
"scheduler": "Planer",
|
||||||
"showMinimapnodes": "MiniMap anzeigen",
|
"showMinimapnodes": "MiniMap anzeigen",
|
||||||
"showLegendNodes": "Feldtyp-Legende anzeigen",
|
|
||||||
"executionStateCompleted": "Erledigt",
|
"executionStateCompleted": "Erledigt",
|
||||||
"downloadWorkflow": "Workflow JSON herunterladen",
|
"downloadWorkflow": "Workflow JSON herunterladen",
|
||||||
"executionStateInProgress": "In Bearbeitung",
|
"executionStateInProgress": "In Bearbeitung",
|
||||||
@@ -1280,7 +1236,6 @@
|
|||||||
"fieldTypesMustMatch": "Feldtypen müssen übereinstimmen",
|
"fieldTypesMustMatch": "Feldtypen müssen übereinstimmen",
|
||||||
"fitViewportNodes": "An Ansichtsgröße anpassen",
|
"fitViewportNodes": "An Ansichtsgröße anpassen",
|
||||||
"loadingNodes": "Lade Nodes...",
|
"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",
|
"fullyContainNodesHelp": "Nodes müssen vollständig innerhalb der Auswahlbox sein, um ausgewählt werden zu können",
|
||||||
"noWorkflow": "Kein Workflow",
|
"noWorkflow": "Kein Workflow",
|
||||||
"executionStateError": "Fehler",
|
"executionStateError": "Fehler",
|
||||||
@@ -1292,9 +1247,7 @@
|
|||||||
"sourceNodeDoesNotExist": "Ungültiger Rand: Quell- / Ausgabe-Knoten {{node}} existiert nicht",
|
"sourceNodeDoesNotExist": "Ungültiger Rand: Quell- / Ausgabe-Knoten {{node}} existiert nicht",
|
||||||
"updateAllNodes": "Update Knoten",
|
"updateAllNodes": "Update Knoten",
|
||||||
"allNodesUpdated": "Alle Knoten aktualisiert",
|
"allNodesUpdated": "Alle Knoten aktualisiert",
|
||||||
"unknownTemplate": "Unbekannte Vorlage",
|
|
||||||
"updateApp": "Update App",
|
"updateApp": "Update App",
|
||||||
"unknownInput": "Unbekannte Eingabe: {{name}}",
|
|
||||||
"unknownNodeType": "Unbekannter Knotentyp",
|
"unknownNodeType": "Unbekannter Knotentyp",
|
||||||
"float": "Kommazahlen",
|
"float": "Kommazahlen",
|
||||||
"enum": "Aufzählung",
|
"enum": "Aufzählung",
|
||||||
@@ -1310,7 +1263,6 @@
|
|||||||
"workflowAuthor": "Autor",
|
"workflowAuthor": "Autor",
|
||||||
"graph": "Graph",
|
"graph": "Graph",
|
||||||
"workflowDescription": "Kurze Beschreibung",
|
"workflowDescription": "Kurze Beschreibung",
|
||||||
"versionUnknown": " Version unbekannt",
|
|
||||||
"workflow": "Arbeitsablauf",
|
"workflow": "Arbeitsablauf",
|
||||||
"noGraph": "Kein Graph",
|
"noGraph": "Kein Graph",
|
||||||
"version": "Version",
|
"version": "Version",
|
||||||
@@ -1328,7 +1280,6 @@
|
|||||||
"unknownErrorValidatingWorkflow": "Unbekannter Fehler beim Validieren des Arbeitsablaufes",
|
"unknownErrorValidatingWorkflow": "Unbekannter Fehler beim Validieren des Arbeitsablaufes",
|
||||||
"inputFieldTypeParseError": "Typ des Eingabefelds {{node}}.{{field}} kann nicht analysiert werden ({{message}})",
|
"inputFieldTypeParseError": "Typ des Eingabefelds {{node}}.{{field}} kann nicht analysiert werden ({{message}})",
|
||||||
"workflowSettings": "Arbeitsablauf Editor Einstellungen",
|
"workflowSettings": "Arbeitsablauf Editor Einstellungen",
|
||||||
"unableToLoadWorkflow": "Arbeitsablauf kann nicht geladen werden",
|
|
||||||
"viewMode": "In linearen Ansicht verwenden",
|
"viewMode": "In linearen Ansicht verwenden",
|
||||||
"unableToValidateWorkflow": "Arbeitsablauf kann nicht validiert werden",
|
"unableToValidateWorkflow": "Arbeitsablauf kann nicht validiert werden",
|
||||||
"outputFieldTypeParseError": "Typ des Ausgabefelds {{node}}.{{field}} kann nicht analysiert werden ({{message}})",
|
"outputFieldTypeParseError": "Typ des Ausgabefelds {{node}}.{{field}} kann nicht analysiert werden ({{message}})",
|
||||||
@@ -1344,7 +1295,6 @@
|
|||||||
"arithmeticSequence": "Arithmetische Folge",
|
"arithmeticSequence": "Arithmetische Folge",
|
||||||
"noBatchGroup": "keine Gruppe",
|
"noBatchGroup": "keine Gruppe",
|
||||||
"generatorNoValues": "leer",
|
"generatorNoValues": "leer",
|
||||||
"generatorLoading": "wird geladen",
|
|
||||||
"generatorLoadFromFile": "Aus Datei laden",
|
"generatorLoadFromFile": "Aus Datei laden",
|
||||||
"showEdgeLabels": "Kantenbeschriftungen anzeigen",
|
"showEdgeLabels": "Kantenbeschriftungen anzeigen",
|
||||||
"downloadWorkflowError": "Fehler beim Herunterladen des Arbeitsablaufs",
|
"downloadWorkflowError": "Fehler beim Herunterladen des Arbeitsablaufs",
|
||||||
@@ -1352,14 +1302,11 @@
|
|||||||
"description": "Beschreibung",
|
"description": "Beschreibung",
|
||||||
"loadWorkflowDesc": "Arbeitsablauf laden?",
|
"loadWorkflowDesc": "Arbeitsablauf laden?",
|
||||||
"loadWorkflowDesc2": "Ihr aktueller Arbeitsablauf enthält nicht gespeicherte Änderungen.",
|
"loadWorkflowDesc2": "Ihr aktueller Arbeitsablauf enthält nicht gespeicherte Änderungen.",
|
||||||
"loadingTemplates": "Lade {{name}}",
|
|
||||||
"missingSourceOrTargetHandle": "Fehlender Quell- oder Zielgriff",
|
"missingSourceOrTargetHandle": "Fehlender Quell- oder Zielgriff",
|
||||||
"missingSourceOrTargetNode": "Fehlender Quell- oder Zielknoten",
|
"missingSourceOrTargetNode": "Fehlender Quell- oder Zielknoten",
|
||||||
"showEdgeLabelsHelp": "Beschriftungen an Kanten anzeigen, um die verknüpften Knoten zu kennzeichnen"
|
"showEdgeLabelsHelp": "Beschriftungen an Kanten anzeigen, um die verknüpften Knoten zu kennzeichnen"
|
||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"enableHrf": "Korrektur für hohe Auflösungen",
|
|
||||||
"upscaleMethod": "Vergrößerungsmethode",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"strength": "Auflösungs-Fix Stärke",
|
"strength": "Auflösungs-Fix Stärke",
|
||||||
"enabled": "Auflösungs-Fix aktiviert",
|
"enabled": "Auflösungs-Fix aktiviert",
|
||||||
@@ -1370,11 +1317,9 @@
|
|||||||
"models": {
|
"models": {
|
||||||
"noMatchingModels": "Keine passenden Modelle",
|
"noMatchingModels": "Keine passenden Modelle",
|
||||||
"loading": "lade",
|
"loading": "lade",
|
||||||
"noMatchingLoRAs": "Keine passenden LoRAs",
|
|
||||||
"noModelsAvailable": "Keine Modelle verfügbar",
|
"noModelsAvailable": "Keine Modelle verfügbar",
|
||||||
"selectModel": "Wählen ein Modell aus",
|
"selectModel": "Wählen ein Modell aus",
|
||||||
"noRefinerModelsInstalled": "Keine SDXL Refiner-Modelle installiert",
|
"noRefinerModelsInstalled": "Keine SDXL Refiner-Modelle installiert",
|
||||||
"noLoRAsInstalled": "Keine LoRAs installiert",
|
|
||||||
"addLora": "LoRA hinzufügen",
|
"addLora": "LoRA hinzufügen",
|
||||||
"defaultVAE": "Standard VAE",
|
"defaultVAE": "Standard VAE",
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
@@ -1404,31 +1349,23 @@
|
|||||||
"workflows": "Arbeitsabläufe",
|
"workflows": "Arbeitsabläufe",
|
||||||
"workflowName": "Arbeitsablauf-Name",
|
"workflowName": "Arbeitsablauf-Name",
|
||||||
"saveWorkflowAs": "Arbeitsablauf speichern als",
|
"saveWorkflowAs": "Arbeitsablauf speichern als",
|
||||||
"searchWorkflows": "Suche Arbeitsabläufe",
|
|
||||||
"newWorkflowCreated": "Neuer Arbeitsablauf erstellt",
|
"newWorkflowCreated": "Neuer Arbeitsablauf erstellt",
|
||||||
"problemSavingWorkflow": "Problem beim Speichern des Arbeitsablaufs",
|
"problemSavingWorkflow": "Problem beim Speichern des Arbeitsablaufs",
|
||||||
"problemLoading": "Problem beim Laden von Arbeitsabläufen",
|
|
||||||
"downloadWorkflow": "Speichern als",
|
"downloadWorkflow": "Speichern als",
|
||||||
"savingWorkflow": "Speichere Arbeitsablauf...",
|
"savingWorkflow": "Speichere Arbeitsablauf...",
|
||||||
"saveWorkflow": "Arbeitsablauf speichern",
|
"saveWorkflow": "Arbeitsablauf speichern",
|
||||||
"noWorkflows": "Keine Arbeitsabläufe",
|
"noWorkflows": "Keine Arbeitsabläufe",
|
||||||
"workflowLibrary": "Bibliothek",
|
"workflowLibrary": "Bibliothek",
|
||||||
"unnamedWorkflow": "Unbenannter Arbeitsablauf",
|
"unnamedWorkflow": "Unbenannter Arbeitsablauf",
|
||||||
"noDescription": "Keine Beschreibung",
|
|
||||||
"clearWorkflowSearchFilter": "Suchfilter zurücksetzen",
|
|
||||||
"workflowEditorMenu": "Arbeitsablauf-Editor Menü",
|
"workflowEditorMenu": "Arbeitsablauf-Editor Menü",
|
||||||
"deleteWorkflow": "Arbeitsablauf löschen",
|
"deleteWorkflow": "Arbeitsablauf löschen",
|
||||||
"workflowSaved": "Arbeitsablauf gespeichert",
|
"workflowSaved": "Arbeitsablauf gespeichert",
|
||||||
"uploadWorkflow": "Aus Datei laden",
|
"uploadWorkflow": "Aus Datei laden",
|
||||||
"openWorkflow": "Arbeitsablauf öffnen",
|
|
||||||
"saveWorkflowToProject": "Arbeitsablauf in Projekt speichern",
|
"saveWorkflowToProject": "Arbeitsablauf in Projekt speichern",
|
||||||
"workflowCleared": "Arbeitsablauf gelöscht",
|
"workflowCleared": "Arbeitsablauf gelöscht",
|
||||||
"loading": "Lade Arbeitsabläufe",
|
"loading": "Lade Arbeitsabläufe",
|
||||||
"name": "Name",
|
"name": "Name",
|
||||||
"ascending": "Aufsteigend",
|
"ascending": "Aufsteigend",
|
||||||
"defaultWorkflows": "Standard Arbeitsabläufe",
|
|
||||||
"userWorkflows": "Benutzer Arbeitsabläufe",
|
|
||||||
"projectWorkflows": "Projekt Arbeitsabläufe",
|
|
||||||
"opened": "Geöffnet",
|
"opened": "Geöffnet",
|
||||||
"loadWorkflow": "Arbeitsablauf $t(common.load)",
|
"loadWorkflow": "Arbeitsablauf $t(common.load)",
|
||||||
"updated": "Aktualisiert",
|
"updated": "Aktualisiert",
|
||||||
@@ -1442,12 +1379,10 @@
|
|||||||
"copyShareLink": "Teilen-Link kopieren",
|
"copyShareLink": "Teilen-Link kopieren",
|
||||||
"download": "Herunterladen",
|
"download": "Herunterladen",
|
||||||
"convertGraph": "Graph konvertieren",
|
"convertGraph": "Graph konvertieren",
|
||||||
"filterByTags": "Nach Tags filtern",
|
|
||||||
"yourWorkflows": "Ihre Arbeitsabläufe",
|
"yourWorkflows": "Ihre Arbeitsabläufe",
|
||||||
"recentlyOpened": "Kürzlich geöffnet"
|
"recentlyOpened": "Kürzlich geöffnet"
|
||||||
},
|
},
|
||||||
"sdxl": {
|
"sdxl": {
|
||||||
"concatPromptStyle": "Verknüpfen von Prompt & Stil",
|
|
||||||
"scheduler": "Planer",
|
"scheduler": "Planer",
|
||||||
"steps": "Schritte"
|
"steps": "Schritte"
|
||||||
},
|
},
|
||||||
@@ -1459,13 +1394,11 @@
|
|||||||
"addPromptTrigger": "Prompt-Trigger hinzufügen",
|
"addPromptTrigger": "Prompt-Trigger hinzufügen",
|
||||||
"compatibleEmbeddings": "Kompatible Einbettungen",
|
"compatibleEmbeddings": "Kompatible Einbettungen",
|
||||||
"replace": "Ersetzen",
|
"replace": "Ersetzen",
|
||||||
"insert": "Einfügen",
|
|
||||||
"discard": "Verwerfen",
|
"discard": "Verwerfen",
|
||||||
"generateFromImage": "Prompt aus Bild generieren",
|
"generateFromImage": "Prompt aus Bild generieren",
|
||||||
"expandCurrentPrompt": "Aktuelle Prompt erweitern",
|
"expandCurrentPrompt": "Aktuelle Prompt erweitern",
|
||||||
"uploadImageForPromptGeneration": "Bild zur Prompt-Generierung hochladen",
|
"uploadImageForPromptGeneration": "Bild zur Prompt-Generierung hochladen",
|
||||||
"expandingPrompt": "Prompt wird erweitert...",
|
"expandingPrompt": "Prompt wird erweitert..."
|
||||||
"resultTitle": "Prompt-Erweiterung abgeschlossen"
|
|
||||||
},
|
},
|
||||||
"ui": {
|
"ui": {
|
||||||
"tabs": {
|
"tabs": {
|
||||||
@@ -1604,8 +1537,6 @@
|
|||||||
"opacity": "Opazität",
|
"opacity": "Opazität",
|
||||||
"removeBookmark": "Lesezeichen entfernen",
|
"removeBookmark": "Lesezeichen entfernen",
|
||||||
"rasterLayer": "Rasterebene",
|
"rasterLayer": "Rasterebene",
|
||||||
"rasterLayers_withCount_visible": "Rasterebenen ({{count}})",
|
|
||||||
"controlLayers_withCount_visible": "Kontroll-Ebenen ({{count}})",
|
|
||||||
"deleteSelected": "Ausgewählte löschen",
|
"deleteSelected": "Ausgewählte löschen",
|
||||||
"newRegionalReferenceImageError": "Problem beim Erstellen eines regionalen Referenzbilds",
|
"newRegionalReferenceImageError": "Problem beim Erstellen eines regionalen Referenzbilds",
|
||||||
"newControlLayerOk": "Kontroll-Ebene erstellt",
|
"newControlLayerOk": "Kontroll-Ebene erstellt",
|
||||||
@@ -1613,10 +1544,8 @@
|
|||||||
"newRasterLayerOk": "Rasterebene erstellt",
|
"newRasterLayerOk": "Rasterebene erstellt",
|
||||||
"moveToFront": "Nach vorne bringen",
|
"moveToFront": "Nach vorne bringen",
|
||||||
"copyToClipboard": "In die Zwischenablage kopieren",
|
"copyToClipboard": "In die Zwischenablage kopieren",
|
||||||
"controlLayers_withCount_hidden": "Kontroll-Ebenen ({{count}} ausgeblendet)",
|
|
||||||
"clearCaches": "Cache leeren",
|
"clearCaches": "Cache leeren",
|
||||||
"controlLayer": "Kontroll-Ebene",
|
"controlLayer": "Kontroll-Ebene",
|
||||||
"rasterLayers_withCount_hidden": "Rasterebenen ({{count}} ausgeblendet)",
|
|
||||||
"transparency": "Transparenz",
|
"transparency": "Transparenz",
|
||||||
"canvas": "Leinwand",
|
"canvas": "Leinwand",
|
||||||
"global": "Global",
|
"global": "Global",
|
||||||
@@ -1639,9 +1568,7 @@
|
|||||||
"weight": "Gewichtung",
|
"weight": "Gewichtung",
|
||||||
"addReferenceImage": "$t(controlLayers.referenceImage) hinzufügen",
|
"addReferenceImage": "$t(controlLayers.referenceImage) hinzufügen",
|
||||||
"addInpaintMask": "$t(controlLayers.inpaintMask) hinzufügen",
|
"addInpaintMask": "$t(controlLayers.inpaintMask) hinzufügen",
|
||||||
"addGlobalReferenceImage": "$t(controlLayers.globalReferenceImage) hinzufügen",
|
|
||||||
"regionalGuidance": "Regionale Führung",
|
"regionalGuidance": "Regionale Führung",
|
||||||
"globalReferenceImages_withCount_visible": "Globale Referenzbilder ({{count}})",
|
|
||||||
"addPositivePrompt": "$t(controlLayers.prompt) hinzufügen",
|
"addPositivePrompt": "$t(controlLayers.prompt) hinzufügen",
|
||||||
"locked": "Gesperrt",
|
"locked": "Gesperrt",
|
||||||
"showHUD": "HUD anzeigen",
|
"showHUD": "HUD anzeigen",
|
||||||
@@ -1649,16 +1576,12 @@
|
|||||||
"addRasterLayer": "$t(controlLayers.rasterLayer) hinzufügen",
|
"addRasterLayer": "$t(controlLayers.rasterLayer) hinzufügen",
|
||||||
"addRegionalGuidance": "$t(controlLayers.regionalGuidance) hinzufügen",
|
"addRegionalGuidance": "$t(controlLayers.regionalGuidance) hinzufügen",
|
||||||
"addControlLayer": "$t(controlLayers.controlLayer) hinzufügen",
|
"addControlLayer": "$t(controlLayers.controlLayer) hinzufügen",
|
||||||
"newCanvasSession": "Neue Leinwand-Sitzung",
|
|
||||||
"replaceLayer": "Ebene ersetzen",
|
"replaceLayer": "Ebene ersetzen",
|
||||||
"newGallerySession": "Neue Galerie-Sitzung",
|
|
||||||
"unlocked": "Entsperrt",
|
"unlocked": "Entsperrt",
|
||||||
"showProgressOnCanvas": "Fortschritt auf Leinwand anzeigen",
|
"showProgressOnCanvas": "Fortschritt auf Leinwand anzeigen",
|
||||||
"controlMode": {
|
"controlMode": {
|
||||||
"balanced": "Ausgewogen"
|
"balanced": "Ausgewogen"
|
||||||
},
|
},
|
||||||
"globalReferenceImages_withCount_hidden": "Globale Referenzbilder ({{count}} ausgeblendet)",
|
|
||||||
"sendToGallery": "An Galerie senden",
|
|
||||||
"stagingArea": {
|
"stagingArea": {
|
||||||
"accept": "Annehmen",
|
"accept": "Annehmen",
|
||||||
"next": "Nächste",
|
"next": "Nächste",
|
||||||
@@ -1666,8 +1589,6 @@
|
|||||||
"discard": "Verwerfen",
|
"discard": "Verwerfen",
|
||||||
"previous": "Vorherige"
|
"previous": "Vorherige"
|
||||||
},
|
},
|
||||||
"regionalGuidance_withCount_visible": "Regionale Führung ({{count}})",
|
|
||||||
"regionalGuidance_withCount_hidden": "Regionale Führung ({{count}} ausgeblendet)",
|
|
||||||
"settings": {
|
"settings": {
|
||||||
"snapToGrid": {
|
"snapToGrid": {
|
||||||
"on": "Ein",
|
"on": "Ein",
|
||||||
@@ -1677,8 +1598,6 @@
|
|||||||
},
|
},
|
||||||
"layer_one": "Ebene",
|
"layer_one": "Ebene",
|
||||||
"layer_other": "Ebenen",
|
"layer_other": "Ebenen",
|
||||||
"layer_withCount_one": "Ebene ({{count}})",
|
|
||||||
"layer_withCount_other": "Ebenen ({{count}})",
|
|
||||||
"fill": {
|
"fill": {
|
||||||
"fillStyle": "Füllstil",
|
"fillStyle": "Füllstil",
|
||||||
"diagonal": "Diagonal",
|
"diagonal": "Diagonal",
|
||||||
|
|||||||
@@ -43,6 +43,8 @@
|
|||||||
"move": "Move",
|
"move": "Move",
|
||||||
"movingImagesToBoard_one": "Moving {{count}} image to board:",
|
"movingImagesToBoard_one": "Moving {{count}} image to board:",
|
||||||
"movingImagesToBoard_other": "Moving {{count}} images to board:",
|
"movingImagesToBoard_other": "Moving {{count}} images to board:",
|
||||||
|
"movingVideosToBoard_one": "Moving {{count}} video to board:",
|
||||||
|
"movingVideosToBoard_other": "Moving {{count}} videos to board:",
|
||||||
"myBoard": "My Board",
|
"myBoard": "My Board",
|
||||||
"noBoards": "No {{boardType}} Boards",
|
"noBoards": "No {{boardType}} Boards",
|
||||||
"noMatching": "No matching Boards",
|
"noMatching": "No matching Boards",
|
||||||
@@ -59,6 +61,8 @@
|
|||||||
"imagesWithCount_other": "{{count}} images",
|
"imagesWithCount_other": "{{count}} images",
|
||||||
"assetsWithCount_one": "{{count}} asset",
|
"assetsWithCount_one": "{{count}} asset",
|
||||||
"assetsWithCount_other": "{{count}} assets",
|
"assetsWithCount_other": "{{count}} assets",
|
||||||
|
"videosWithCount_one": "{{count}} video",
|
||||||
|
"videosWithCount_other": "{{count}} videos",
|
||||||
"updateBoardError": "Error updating board"
|
"updateBoardError": "Error updating board"
|
||||||
},
|
},
|
||||||
"accordions": {
|
"accordions": {
|
||||||
@@ -100,6 +104,7 @@
|
|||||||
"copy": "Copy",
|
"copy": "Copy",
|
||||||
"copyError": "$t(gallery.copy) Error",
|
"copyError": "$t(gallery.copy) Error",
|
||||||
"clipboard": "Clipboard",
|
"clipboard": "Clipboard",
|
||||||
|
"crop": "Crop",
|
||||||
"on": "On",
|
"on": "On",
|
||||||
"off": "Off",
|
"off": "Off",
|
||||||
"or": "or",
|
"or": "or",
|
||||||
@@ -238,7 +243,10 @@
|
|||||||
"resultSubtitle": "Choose how to handle the expanded prompt:",
|
"resultSubtitle": "Choose how to handle the expanded prompt:",
|
||||||
"replace": "Replace",
|
"replace": "Replace",
|
||||||
"insert": "Insert",
|
"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": "Queue",
|
"queue": "Queue",
|
||||||
@@ -294,7 +302,7 @@
|
|||||||
"completedIn": "Completed in",
|
"completedIn": "Completed in",
|
||||||
"batch": "Batch",
|
"batch": "Batch",
|
||||||
"origin": "Origin",
|
"origin": "Origin",
|
||||||
"destination": "Destination",
|
"destination": "Dest",
|
||||||
"upscaling": "Upscaling",
|
"upscaling": "Upscaling",
|
||||||
"canvas": "Canvas",
|
"canvas": "Canvas",
|
||||||
"generation": "Generation",
|
"generation": "Generation",
|
||||||
@@ -320,7 +328,13 @@
|
|||||||
"iterations_other": "Iterations",
|
"iterations_other": "Iterations",
|
||||||
"generations_one": "Generation",
|
"generations_one": "Generation",
|
||||||
"generations_other": "Generations",
|
"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": {
|
||||||
"invocationCache": "Invocation Cache",
|
"invocationCache": "Invocation Cache",
|
||||||
@@ -361,6 +375,9 @@
|
|||||||
"deleteImage_one": "Delete Image",
|
"deleteImage_one": "Delete Image",
|
||||||
"deleteImage_other": "Delete {{count}} Images",
|
"deleteImage_other": "Delete {{count}} Images",
|
||||||
"deleteImagePermanent": "Deleted images cannot be restored.",
|
"deleteImagePermanent": "Deleted images cannot be restored.",
|
||||||
|
"deleteVideo_one": "Delete Video",
|
||||||
|
"deleteVideo_other": "Delete {{count}} Videos",
|
||||||
|
"deleteVideoPermanent": "Deleted videos cannot be restored.",
|
||||||
"displayBoardSearch": "Board Search",
|
"displayBoardSearch": "Board Search",
|
||||||
"displaySearch": "Image Search",
|
"displaySearch": "Image Search",
|
||||||
"download": "Download",
|
"download": "Download",
|
||||||
@@ -380,9 +397,10 @@
|
|||||||
"sortDirection": "Sort Direction",
|
"sortDirection": "Sort Direction",
|
||||||
"showStarredImagesFirst": "Show Starred Images First",
|
"showStarredImagesFirst": "Show Starred Images First",
|
||||||
"noImageSelected": "No Image Selected",
|
"noImageSelected": "No Image Selected",
|
||||||
|
"noVideoSelected": "No Video Selected",
|
||||||
"noImagesInGallery": "No Images to Display",
|
"noImagesInGallery": "No Images to Display",
|
||||||
"starImage": "Star Image",
|
"starImage": "Star",
|
||||||
"unstarImage": "Unstar Image",
|
"unstarImage": "Unstar",
|
||||||
"unableToLoad": "Unable to load Gallery",
|
"unableToLoad": "Unable to load Gallery",
|
||||||
"deleteSelection": "Delete Selection",
|
"deleteSelection": "Delete Selection",
|
||||||
"downloadSelection": "Download Selection",
|
"downloadSelection": "Download Selection",
|
||||||
@@ -411,7 +429,9 @@
|
|||||||
"openViewer": "Open Viewer",
|
"openViewer": "Open Viewer",
|
||||||
"closeViewer": "Close Viewer",
|
"closeViewer": "Close Viewer",
|
||||||
"move": "Move",
|
"move": "Move",
|
||||||
"useForPromptGeneration": "Use for Prompt Generation"
|
"useForPromptGeneration": "Use for Prompt Generation",
|
||||||
|
"videos": "Videos",
|
||||||
|
"videosTab": "Videos you've created and saved within Invoke."
|
||||||
},
|
},
|
||||||
"hotkeys": {
|
"hotkeys": {
|
||||||
"hotkeys": "Hotkeys",
|
"hotkeys": "Hotkeys",
|
||||||
@@ -456,10 +476,22 @@
|
|||||||
"title": "Select the Queue Tab",
|
"title": "Select the Queue Tab",
|
||||||
"desc": "Selects the Queue tab."
|
"desc": "Selects the Queue tab."
|
||||||
},
|
},
|
||||||
|
"selectVideoTab": {
|
||||||
|
"title": "Select the Video Tab",
|
||||||
|
"desc": "Selects the Video tab."
|
||||||
|
},
|
||||||
"focusPrompt": {
|
"focusPrompt": {
|
||||||
"title": "Focus Prompt",
|
"title": "Focus Prompt",
|
||||||
"desc": "Move cursor focus to the positive 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": {
|
"toggleLeftPanel": {
|
||||||
"title": "Toggle Left Panel",
|
"title": "Toggle Left Panel",
|
||||||
"desc": "Show or hide the left panel."
|
"desc": "Show or hide the left panel."
|
||||||
@@ -482,6 +514,9 @@
|
|||||||
"key": "1"
|
"key": "1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"video": {
|
||||||
|
"title": "Video"
|
||||||
|
},
|
||||||
"canvas": {
|
"canvas": {
|
||||||
"title": "Canvas",
|
"title": "Canvas",
|
||||||
"selectBrushTool": {
|
"selectBrushTool": {
|
||||||
@@ -572,9 +607,13 @@
|
|||||||
"title": "Prev Layer",
|
"title": "Prev Layer",
|
||||||
"desc": "Select the previous layer in the list."
|
"desc": "Select the previous layer in the list."
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
"setFillColorsToDefault": {
|
||||||
"title": "Set Color to White",
|
"title": "Set Colors to Default",
|
||||||
"desc": "Set the current tool color to white."
|
"desc": "Set the current tool colors to default."
|
||||||
|
},
|
||||||
|
"toggleFillColor": {
|
||||||
|
"title": "Toggle Fill Color",
|
||||||
|
"desc": "Toggle the current tool fill color."
|
||||||
},
|
},
|
||||||
"filterSelected": {
|
"filterSelected": {
|
||||||
"title": "Filter",
|
"title": "Filter",
|
||||||
@@ -771,6 +810,9 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"lora": {
|
||||||
|
"weight": "Weight"
|
||||||
|
},
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"allPrompts": "All Prompts",
|
"allPrompts": "All Prompts",
|
||||||
"cfgScale": "CFG scale",
|
"cfgScale": "CFG scale",
|
||||||
@@ -781,11 +823,13 @@
|
|||||||
"guidance": "Guidance",
|
"guidance": "Guidance",
|
||||||
"height": "Height",
|
"height": "Height",
|
||||||
"imageDetails": "Image Details",
|
"imageDetails": "Image Details",
|
||||||
|
"videoDetails": "Video Details",
|
||||||
"imageDimensions": "Image Dimensions",
|
"imageDimensions": "Image Dimensions",
|
||||||
"metadata": "Metadata",
|
"metadata": "Metadata",
|
||||||
"model": "Model",
|
"model": "Model",
|
||||||
"negativePrompt": "Negative Prompt",
|
"negativePrompt": "Negative Prompt",
|
||||||
"noImageDetails": "No image details found",
|
"noImageDetails": "No image details found",
|
||||||
|
"noVideoDetails": "No video details found",
|
||||||
"noMetaData": "No metadata found",
|
"noMetaData": "No metadata found",
|
||||||
"noRecallParameters": "No parameters to recall found",
|
"noRecallParameters": "No parameters to recall found",
|
||||||
"parameterSet": "Parameter {{parameter}} set",
|
"parameterSet": "Parameter {{parameter}} set",
|
||||||
@@ -803,7 +847,11 @@
|
|||||||
"vae": "VAE",
|
"vae": "VAE",
|
||||||
"width": "Width",
|
"width": "Width",
|
||||||
"workflow": "Workflow",
|
"workflow": "Workflow",
|
||||||
"canvasV2Metadata": "Canvas Layers"
|
"canvasV2Metadata": "Canvas Layers",
|
||||||
|
"videoModel": "Model",
|
||||||
|
"videoDuration": "Duration",
|
||||||
|
"videoAspectRatio": "Aspect Ratio",
|
||||||
|
"videoResolution": "Resolution"
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
"active": "active",
|
"active": "active",
|
||||||
@@ -1189,6 +1237,7 @@
|
|||||||
},
|
},
|
||||||
"parameters": {
|
"parameters": {
|
||||||
"aspect": "Aspect",
|
"aspect": "Aspect",
|
||||||
|
"duration": "Duration",
|
||||||
"lockAspectRatio": "Lock Aspect Ratio",
|
"lockAspectRatio": "Lock Aspect Ratio",
|
||||||
"swapDimensions": "Swap Dimensions",
|
"swapDimensions": "Swap Dimensions",
|
||||||
"setToOptimalSize": "Optimize size for model",
|
"setToOptimalSize": "Optimize size for model",
|
||||||
@@ -1213,9 +1262,15 @@
|
|||||||
"height": "Height",
|
"height": "Height",
|
||||||
"imageFit": "Fit Initial Image To Output Size",
|
"imageFit": "Fit Initial Image To Output Size",
|
||||||
"images": "Images",
|
"images": "Images",
|
||||||
|
"images_withCount_one": "Image",
|
||||||
|
"images_withCount_other": "Images",
|
||||||
|
"videos_withCount_one": "Video",
|
||||||
|
"videos_withCount_other": "Videos",
|
||||||
"infillMethod": "Infill Method",
|
"infillMethod": "Infill Method",
|
||||||
"infillColorValue": "Fill Color",
|
"infillColorValue": "Fill Color",
|
||||||
"info": "Info",
|
"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": {
|
"invoke": {
|
||||||
"addingImagesTo": "Adding images to",
|
"addingImagesTo": "Adding images to",
|
||||||
"modelDisabledForTrial": "Generating with {{modelName}} is not available on trial accounts. Visit your account settings to upgrade.",
|
"modelDisabledForTrial": "Generating with {{modelName}} is not available on trial accounts. Visit your account settings to upgrade.",
|
||||||
@@ -1239,6 +1294,7 @@
|
|||||||
"batchNodeCollectionSizeMismatchNoGroupId": "Batch group collection size mismatch",
|
"batchNodeCollectionSizeMismatchNoGroupId": "Batch group collection size mismatch",
|
||||||
"batchNodeCollectionSizeMismatch": "Collection size mismatch on Batch {{batchGroupId}}",
|
"batchNodeCollectionSizeMismatch": "Collection size mismatch on Batch {{batchGroupId}}",
|
||||||
"noModelSelected": "No model selected",
|
"noModelSelected": "No model selected",
|
||||||
|
"noStartingFrameImage": "No starting frame image",
|
||||||
"noT5EncoderModelSelected": "No T5 Encoder model selected for FLUX generation",
|
"noT5EncoderModelSelected": "No T5 Encoder model selected for FLUX generation",
|
||||||
"noFLUXVAEModelSelected": "No VAE model selected for FLUX generation",
|
"noFLUXVAEModelSelected": "No VAE model selected for FLUX generation",
|
||||||
"noCLIPEmbedModelSelected": "No CLIP Embed model selected for FLUX generation",
|
"noCLIPEmbedModelSelected": "No CLIP Embed model selected for FLUX generation",
|
||||||
@@ -1251,7 +1307,7 @@
|
|||||||
"modelIncompatibleScaledBboxWidth": "Scaled bbox width is {{width}} but {{model}} requires multiple of {{multiple}}",
|
"modelIncompatibleScaledBboxWidth": "Scaled bbox width is {{width}} but {{model}} requires multiple of {{multiple}}",
|
||||||
"modelIncompatibleScaledBboxHeight": "Scaled bbox height is {{height}} but {{model}} requires multiple of {{multiple}}",
|
"modelIncompatibleScaledBboxHeight": "Scaled bbox height is {{height}} but {{model}} requires multiple of {{multiple}}",
|
||||||
"fluxModelMultipleControlLoRAs": "Can only use 1 Control LoRA at a time",
|
"fluxModelMultipleControlLoRAs": "Can only use 1 Control LoRA at a time",
|
||||||
"fluxKontextMultipleReferenceImages": "Can only use 1 Reference Image at a time with FLUX Kontext via BFL API",
|
"incompatibleLoRAs": "Incompatible LoRA(s) added",
|
||||||
"canvasIsFiltering": "Canvas is busy (filtering)",
|
"canvasIsFiltering": "Canvas is busy (filtering)",
|
||||||
"canvasIsTransforming": "Canvas is busy (transforming)",
|
"canvasIsTransforming": "Canvas is busy (transforming)",
|
||||||
"canvasIsRasterizing": "Canvas is busy (rasterizing)",
|
"canvasIsRasterizing": "Canvas is busy (rasterizing)",
|
||||||
@@ -1261,7 +1317,8 @@
|
|||||||
"noNodesInGraph": "No nodes in graph",
|
"noNodesInGraph": "No nodes in graph",
|
||||||
"systemDisconnected": "System disconnected",
|
"systemDisconnected": "System disconnected",
|
||||||
"promptExpansionPending": "Prompt expansion in progress",
|
"promptExpansionPending": "Prompt expansion in progress",
|
||||||
"promptExpansionResultPending": "Please accept or discard your prompt expansion result"
|
"promptExpansionResultPending": "Please accept or discard your prompt expansion result",
|
||||||
|
"videoIsDisabled": "Video generation is not enabled for {{accountType}} accounts."
|
||||||
},
|
},
|
||||||
"maskBlur": "Mask Blur",
|
"maskBlur": "Mask Blur",
|
||||||
"negativePromptPlaceholder": "Negative Prompt",
|
"negativePromptPlaceholder": "Negative Prompt",
|
||||||
@@ -1279,9 +1336,11 @@
|
|||||||
"seamlessXAxis": "Seamless X Axis",
|
"seamlessXAxis": "Seamless X Axis",
|
||||||
"seamlessYAxis": "Seamless Y Axis",
|
"seamlessYAxis": "Seamless Y Axis",
|
||||||
"seed": "Seed",
|
"seed": "Seed",
|
||||||
|
"videoActions": "Video Actions",
|
||||||
"imageActions": "Image Actions",
|
"imageActions": "Image Actions",
|
||||||
"sendToCanvas": "Send To Canvas",
|
"sendToCanvas": "Send To Canvas",
|
||||||
"sendToUpscale": "Send To Upscale",
|
"sendToUpscale": "Send To Upscale",
|
||||||
|
"sendToVideo": "Send To Video",
|
||||||
"showOptionsPanel": "Show Side Panel (O or T)",
|
"showOptionsPanel": "Show Side Panel (O or T)",
|
||||||
"shuffle": "Shuffle Seed",
|
"shuffle": "Shuffle Seed",
|
||||||
"steps": "Steps",
|
"steps": "Steps",
|
||||||
@@ -1293,6 +1352,7 @@
|
|||||||
"postProcessing": "Post-Processing (Shift + U)",
|
"postProcessing": "Post-Processing (Shift + U)",
|
||||||
"processImage": "Process Image",
|
"processImage": "Process Image",
|
||||||
"upscaling": "Upscaling",
|
"upscaling": "Upscaling",
|
||||||
|
"video": "Video",
|
||||||
"useAll": "Use All",
|
"useAll": "Use All",
|
||||||
"useSize": "Use Size",
|
"useSize": "Use Size",
|
||||||
"useCpuNoise": "Use CPU Noise",
|
"useCpuNoise": "Use CPU Noise",
|
||||||
@@ -1304,6 +1364,7 @@
|
|||||||
"gaussianBlur": "Gaussian Blur",
|
"gaussianBlur": "Gaussian Blur",
|
||||||
"boxBlur": "Box Blur",
|
"boxBlur": "Box Blur",
|
||||||
"staged": "Staged",
|
"staged": "Staged",
|
||||||
|
"resolution": "Resolution",
|
||||||
"modelDisabledForTrial": "Generating with {{modelName}} is not available on trial accounts. Visit your <LinkComponent>account settings</LinkComponent> to upgrade."
|
"modelDisabledForTrial": "Generating with {{modelName}} is not available on trial accounts. Visit your <LinkComponent>account settings</LinkComponent> to upgrade."
|
||||||
},
|
},
|
||||||
"dynamicPrompts": {
|
"dynamicPrompts": {
|
||||||
@@ -2035,6 +2096,24 @@
|
|||||||
"pullBboxIntoLayerError": "Problem Pulling BBox Into Layer",
|
"pullBboxIntoLayerError": "Problem Pulling BBox Into Layer",
|
||||||
"pullBboxIntoReferenceImageOk": "Bbox Pulled Into ReferenceImage",
|
"pullBboxIntoReferenceImageOk": "Bbox Pulled Into ReferenceImage",
|
||||||
"pullBboxIntoReferenceImageError": "Problem Pulling BBox 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",
|
"regionIsEmpty": "Selected region is empty",
|
||||||
"mergeVisible": "Merge Visible",
|
"mergeVisible": "Merge Visible",
|
||||||
"mergeDown": "Merge Down",
|
"mergeDown": "Merge Down",
|
||||||
@@ -2233,6 +2312,8 @@
|
|||||||
},
|
},
|
||||||
"fill": {
|
"fill": {
|
||||||
"fillColor": "Fill Color",
|
"fillColor": "Fill Color",
|
||||||
|
"bgFillColor": "Background Color",
|
||||||
|
"fgFillColor": "Foreground Color",
|
||||||
"fillStyle": "Fill Style",
|
"fillStyle": "Fill Style",
|
||||||
"solid": "Solid",
|
"solid": "Solid",
|
||||||
"grid": "Grid",
|
"grid": "Grid",
|
||||||
@@ -2404,12 +2485,21 @@
|
|||||||
"saveAs": "Save As",
|
"saveAs": "Save As",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"process": "Process",
|
"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.",
|
"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.",
|
||||||
"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.",
|
"visualModeDesc": "Visual mode uses box and point inputs to select an object.",
|
||||||
"help3": "Invert the selection to select everything except the target 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",
|
"clickToAdd": "Click on the layer to add a point",
|
||||||
"dragToMove": "Drag a point to move it",
|
"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": {
|
"settings": {
|
||||||
"snapToGrid": {
|
"snapToGrid": {
|
||||||
@@ -2565,19 +2655,30 @@
|
|||||||
"queue": "Queue",
|
"queue": "Queue",
|
||||||
"upscaling": "Upscaling",
|
"upscaling": "Upscaling",
|
||||||
"upscalingTab": "$t(ui.tabs.upscaling) $t(common.tab)",
|
"upscalingTab": "$t(ui.tabs.upscaling) $t(common.tab)",
|
||||||
|
"video": "Video",
|
||||||
"gallery": "Gallery"
|
"gallery": "Gallery"
|
||||||
},
|
},
|
||||||
"panels": {
|
"panels": {
|
||||||
"launchpad": "Launchpad",
|
"launchpad": "Launchpad",
|
||||||
"workflowEditor": "Workflow Editor",
|
"workflowEditor": "Workflow Editor",
|
||||||
"imageViewer": "Image Viewer",
|
"imageViewer": "Viewer",
|
||||||
"canvas": "Canvas"
|
"canvas": "Canvas",
|
||||||
|
"video": "Video"
|
||||||
},
|
},
|
||||||
"launchpad": {
|
"launchpad": {
|
||||||
"workflowsTitle": "Go deep with Workflows.",
|
"workflowsTitle": "Go deep with Workflows.",
|
||||||
"upscalingTitle": "Upscale and add detail.",
|
"upscalingTitle": "Upscale and add detail.",
|
||||||
"canvasTitle": "Edit and refine on Canvas.",
|
"canvasTitle": "Edit and refine on Canvas.",
|
||||||
"generateTitle": "Generate images from text prompts.",
|
"generateTitle": "Generate images from text prompts.",
|
||||||
|
"videoTitle": "Generate videos from text prompts.",
|
||||||
|
"video": {
|
||||||
|
"startingFrameCalloutTitle": "Add a Starting Frame",
|
||||||
|
"startingFrameCalloutDesc": "Add an image to control the first frame of your video."
|
||||||
|
},
|
||||||
|
"addStartingFrame": {
|
||||||
|
"title": "Add a Starting Frame",
|
||||||
|
"description": "Add an image to control the first frame of your video."
|
||||||
|
},
|
||||||
"modelGuideText": "Want to learn what prompts work best for each model?",
|
"modelGuideText": "Want to learn what prompts work best for each model?",
|
||||||
"modelGuideLink": "Check out our Model Guide.",
|
"modelGuideLink": "Check out our Model Guide.",
|
||||||
"createNewWorkflowFromScratch": "Create a new Workflow from scratch",
|
"createNewWorkflowFromScratch": "Create a new Workflow from scratch",
|
||||||
@@ -2652,6 +2753,10 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"video": {
|
||||||
|
"noVideoSelected": "No video selected",
|
||||||
|
"selectFromGallery": "Select a video from the gallery to play"
|
||||||
|
},
|
||||||
"system": {
|
"system": {
|
||||||
"enableLogging": "Enable Logging",
|
"enableLogging": "Enable Logging",
|
||||||
"logLevel": {
|
"logLevel": {
|
||||||
@@ -2689,8 +2794,9 @@
|
|||||||
"whatsNew": {
|
"whatsNew": {
|
||||||
"whatsNewInInvoke": "What's New in Invoke",
|
"whatsNewInInvoke": "What's New in Invoke",
|
||||||
"items": [
|
"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",
|
"Select Object v2: Improved object selection with point and box inputs or text prompts.",
|
||||||
"Workflows: Add a Shuffle button to number input fields"
|
"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",
|
"readReleaseNotes": "Read Release Notes",
|
||||||
"watchRecentReleaseVideos": "Watch Recent Release Videos",
|
"watchRecentReleaseVideos": "Watch Recent Release Videos",
|
||||||
|
|||||||
@@ -47,11 +47,8 @@
|
|||||||
"editor": "Editor",
|
"editor": "Editor",
|
||||||
"orderBy": "Ordenar por",
|
"orderBy": "Ordenar por",
|
||||||
"file": "Archivo",
|
"file": "Archivo",
|
||||||
"goTo": "Ir a",
|
|
||||||
"imageFailedToLoad": "No se puede cargar la imagen",
|
|
||||||
"saveAs": "Guardar Como",
|
"saveAs": "Guardar Como",
|
||||||
"somethingWentWrong": "Algo salió mal",
|
"somethingWentWrong": "Algo salió mal",
|
||||||
"nextPage": "Página Siguiente",
|
|
||||||
"selected": "Seleccionado",
|
"selected": "Seleccionado",
|
||||||
"tab": "Tabulador",
|
"tab": "Tabulador",
|
||||||
"positivePrompt": "Prompt Positivo",
|
"positivePrompt": "Prompt Positivo",
|
||||||
@@ -61,7 +58,6 @@
|
|||||||
"unknown": "Desconocido",
|
"unknown": "Desconocido",
|
||||||
"input": "Entrada",
|
"input": "Entrada",
|
||||||
"template": "Plantilla",
|
"template": "Plantilla",
|
||||||
"prevPage": "Página Anterior",
|
|
||||||
"red": "Rojo",
|
"red": "Rojo",
|
||||||
"alpha": "Transparencia",
|
"alpha": "Transparencia",
|
||||||
"outputs": "Resultados",
|
"outputs": "Resultados",
|
||||||
@@ -94,8 +90,6 @@
|
|||||||
"edit": "Editar",
|
"edit": "Editar",
|
||||||
"safetensors": "Safetensors",
|
"safetensors": "Safetensors",
|
||||||
"toResolve": "Para resolver",
|
"toResolve": "Para resolver",
|
||||||
"localSystem": "Sistema local",
|
|
||||||
"notInstalled": "No $t(common.installed)",
|
|
||||||
"outpaint": "outpaint",
|
"outpaint": "outpaint",
|
||||||
"simple": "Sencillo",
|
"simple": "Sencillo",
|
||||||
"close": "Cerrar"
|
"close": "Cerrar"
|
||||||
@@ -104,7 +98,6 @@
|
|||||||
"galleryImageSize": "Tamaño de la imagen",
|
"galleryImageSize": "Tamaño de la imagen",
|
||||||
"gallerySettings": "Ajustes de la galería",
|
"gallerySettings": "Ajustes de la galería",
|
||||||
"autoSwitchNewImages": "Auto seleccionar Imágenes nuevas",
|
"autoSwitchNewImages": "Auto seleccionar Imágenes nuevas",
|
||||||
"noImagesInGallery": "No hay imágenes para mostrar",
|
|
||||||
"deleteImage_one": "Eliminar Imagen",
|
"deleteImage_one": "Eliminar Imagen",
|
||||||
"deleteImage_many": "Eliminar {{count}} Imágenes",
|
"deleteImage_many": "Eliminar {{count}} Imágenes",
|
||||||
"deleteImage_other": "Eliminar {{count}} Imágenes",
|
"deleteImage_other": "Eliminar {{count}} Imágenes",
|
||||||
@@ -118,9 +111,7 @@
|
|||||||
"selectForCompare": "Seleccionar para comparar",
|
"selectForCompare": "Seleccionar para comparar",
|
||||||
"alwaysShowImageSizeBadge": "Mostrar siempre las dimensiones de la imagen",
|
"alwaysShowImageSizeBadge": "Mostrar siempre las dimensiones de la imagen",
|
||||||
"currentlyInUse": "Esta imagen se utiliza actualmente con las siguientes funciones:",
|
"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",
|
"selectAllOnPage": "Seleccionar todo en la página",
|
||||||
"selectAnImageToCompare": "Seleccione una imagen para comparar",
|
|
||||||
"bulkDownloadFailed": "Error en la descarga",
|
"bulkDownloadFailed": "Error en la descarga",
|
||||||
"compareHelp2": "Presione <Kbd> M </Kbd> para recorrer los modos de comparación.",
|
"compareHelp2": "Presione <Kbd> M </Kbd> para recorrer los modos de comparación.",
|
||||||
"move": "Mover",
|
"move": "Mover",
|
||||||
@@ -145,7 +136,6 @@
|
|||||||
"exitBoardSearch": "Finalizar búsqueda",
|
"exitBoardSearch": "Finalizar búsqueda",
|
||||||
"exitSearch": "Salir de la búsqueda de imágenes",
|
"exitSearch": "Salir de la búsqueda de imágenes",
|
||||||
"featuresWillReset": "Si elimina esta imagen, dichas funciones se restablecerán inmediatamente.",
|
"featuresWillReset": "Si elimina esta imagen, dichas funciones se restablecerán inmediatamente.",
|
||||||
"jump": "Omitir",
|
|
||||||
"loading": "Cargando",
|
"loading": "Cargando",
|
||||||
"newestFirst": "La más nueva primero",
|
"newestFirst": "La más nueva primero",
|
||||||
"unstarImage": "Dejar de ser favorita",
|
"unstarImage": "Dejar de ser favorita",
|
||||||
@@ -163,9 +153,7 @@
|
|||||||
"boardsSettings": "Ajustes de los tableros",
|
"boardsSettings": "Ajustes de los tableros",
|
||||||
"imagesSettings": "Configuración de imágenes de la galería",
|
"imagesSettings": "Configuración de imágenes de la galería",
|
||||||
"compareHelp3": "Presione <Kbd> C </Kbd> para intercambiar las imágenes comparadas.",
|
"compareHelp3": "Presione <Kbd> C </Kbd> para intercambiar las imágenes comparadas.",
|
||||||
"showArchivedBoards": "Mostrar paneles archivados",
|
"showArchivedBoards": "Mostrar paneles archivados"
|
||||||
"closeViewer": "Cerrar visor",
|
|
||||||
"openViewer": "Abrir visor"
|
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
"modelManager": "Gestor de Modelos",
|
"modelManager": "Gestor de Modelos",
|
||||||
@@ -239,12 +227,10 @@
|
|||||||
"scaledHeight": "Alto escalado",
|
"scaledHeight": "Alto escalado",
|
||||||
"infillMethod": "Método de relleno",
|
"infillMethod": "Método de relleno",
|
||||||
"tileSize": "Tamaño del mosaico",
|
"tileSize": "Tamaño del mosaico",
|
||||||
"downloadImage": "Descargar imagen",
|
|
||||||
"usePrompt": "Usar Entrada",
|
"usePrompt": "Usar Entrada",
|
||||||
"useSeed": "Usar Semilla",
|
"useSeed": "Usar Semilla",
|
||||||
"useAll": "Usar Todo",
|
"useAll": "Usar Todo",
|
||||||
"info": "Información",
|
"info": "Información",
|
||||||
"showOptionsPanel": "Mostrar panel lateral (O o T)",
|
|
||||||
"symmetry": "Simetría",
|
"symmetry": "Simetría",
|
||||||
"copyImage": "Copiar la imagen",
|
"copyImage": "Copiar la imagen",
|
||||||
"general": "General",
|
"general": "General",
|
||||||
@@ -323,8 +309,6 @@
|
|||||||
"hideMinimapnodes": "Ocultar el minimapa",
|
"hideMinimapnodes": "Ocultar el minimapa",
|
||||||
"fitViewportNodes": "Ajustar la vista",
|
"fitViewportNodes": "Ajustar la vista",
|
||||||
"zoomOutNodes": "Alejar",
|
"zoomOutNodes": "Alejar",
|
||||||
"hideLegendNodes": "Ocultar la leyenda del tipo de campo",
|
|
||||||
"showLegendNodes": "Mostrar la leyenda del tipo de campo",
|
|
||||||
"showMinimapnodes": "Mostrar el minimapa",
|
"showMinimapnodes": "Mostrar el minimapa",
|
||||||
"reloadNodeTemplates": "Recargar las plantillas de nodos",
|
"reloadNodeTemplates": "Recargar las plantillas de nodos",
|
||||||
"loadWorkflow": "Cargar el flujo de trabajo",
|
"loadWorkflow": "Cargar el flujo de trabajo",
|
||||||
@@ -361,7 +345,6 @@
|
|||||||
"assetsWithCount_one": "{{count}} activo",
|
"assetsWithCount_one": "{{count}} activo",
|
||||||
"assetsWithCount_many": "{{count}} activos",
|
"assetsWithCount_many": "{{count}} activos",
|
||||||
"assetsWithCount_other": "{{count}} activos",
|
"assetsWithCount_other": "{{count}} activos",
|
||||||
"hideBoards": "Ocultar paneles",
|
|
||||||
"addPrivateBoard": "Agregar un panel privado",
|
"addPrivateBoard": "Agregar un panel privado",
|
||||||
"addSharedBoard": "Añadir panel compartido",
|
"addSharedBoard": "Añadir panel compartido",
|
||||||
"boards": "Paneles",
|
"boards": "Paneles",
|
||||||
@@ -372,7 +355,6 @@
|
|||||||
"noBoards": "No hay paneles {{boardType}}",
|
"noBoards": "No hay paneles {{boardType}}",
|
||||||
"shared": "Paneles compartidos",
|
"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.",
|
"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",
|
"private": "Paneles privados",
|
||||||
"updateBoardError": "No se pudo actualizar el panel"
|
"updateBoardError": "No se pudo actualizar el panel"
|
||||||
},
|
},
|
||||||
@@ -461,7 +443,6 @@
|
|||||||
"other": "Otro",
|
"other": "Otro",
|
||||||
"queueFront": "Añadir al principio de la cola",
|
"queueFront": "Añadir al principio de la cola",
|
||||||
"gallery": "Galería",
|
"gallery": "Galería",
|
||||||
"batchFieldValues": "Valores de procesamiento por lotes",
|
|
||||||
"session": "Sesión",
|
"session": "Sesión",
|
||||||
"notReady": "La cola aún no está lista",
|
"notReady": "La cola aún no está lista",
|
||||||
"graphQueued": "Gráfico en cola",
|
"graphQueued": "Gráfico en cola",
|
||||||
@@ -494,15 +475,11 @@
|
|||||||
"layer_one": "Capa",
|
"layer_one": "Capa",
|
||||||
"layer_many": "Capas",
|
"layer_many": "Capas",
|
||||||
"layer_other": "Capas",
|
"layer_other": "Capas",
|
||||||
"layer_withCount_one": "({{count}}) capa",
|
|
||||||
"layer_withCount_many": "({{count}}) capas",
|
|
||||||
"layer_withCount_other": "({{count}}) capas",
|
|
||||||
"copyToClipboard": "Copiar al portapapeles"
|
"copyToClipboard": "Copiar al portapapeles"
|
||||||
},
|
},
|
||||||
"whatsNew": {
|
"whatsNew": {
|
||||||
"readReleaseNotes": "Leer las notas de la versión",
|
"readReleaseNotes": "Leer las notas de la versión",
|
||||||
"watchRecentReleaseVideos": "Ver videos de versiones recientes",
|
"watchRecentReleaseVideos": "Ver videos de versiones recientes",
|
||||||
"watchUiUpdatesOverview": "Descripción general de las actualizaciones de la interfaz de usuario de Watch",
|
|
||||||
"whatsNewInInvoke": "Novedades en Invoke",
|
"whatsNewInInvoke": "Novedades en Invoke",
|
||||||
"items": [
|
"items": [
|
||||||
"<StrongComponent>SD 3.5</StrongComponent>: compatibilidad con SD 3.5 Medium y Large."
|
"<StrongComponent>SD 3.5</StrongComponent>: compatibilidad con SD 3.5 Medium y Large."
|
||||||
@@ -527,13 +504,11 @@
|
|||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"hrf": "Solución de alta resolución",
|
"hrf": "Solución de alta resolución",
|
||||||
"enableHrf": "Activar corrección de alta resolución",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"enabled": "Corrección de alta resolución activada",
|
"enabled": "Corrección de alta resolución activada",
|
||||||
"strength": "Forzar la corrección de alta resolución",
|
"strength": "Forzar la corrección de alta resolución",
|
||||||
"method": "Método de corrección de alta resolución"
|
"method": "Método de corrección de alta resolución"
|
||||||
},
|
}
|
||||||
"upscaleMethod": "Método de expansión"
|
|
||||||
},
|
},
|
||||||
"prompt": {
|
"prompt": {
|
||||||
"addPromptTrigger": "Añadir activador de los avisos",
|
"addPromptTrigger": "Añadir activador de los avisos",
|
||||||
@@ -591,10 +566,6 @@
|
|||||||
"title": "Ajustar capas al lienzo",
|
"title": "Ajustar capas al lienzo",
|
||||||
"desc": "Escala y posiciona la vista para que se ajuste a todas las capas visibles."
|
"desc": "Escala y posiciona la vista para que se ajuste a todas las capas visibles."
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "Establecer color en blanco",
|
|
||||||
"desc": "Establece el color actual de la herramienta en blanco."
|
|
||||||
},
|
|
||||||
"resetSelected": {
|
"resetSelected": {
|
||||||
"title": "Restablecer capa",
|
"title": "Restablecer capa",
|
||||||
"desc": "Restablecer la capa seleccionada. Solo se aplica a Máscara de retoque y Guía regional."
|
"desc": "Restablecer la capa seleccionada. Solo se aplica a Máscara de retoque y Guía regional."
|
||||||
@@ -868,10 +839,8 @@
|
|||||||
"seed": "Semilla",
|
"seed": "Semilla",
|
||||||
"strength": "Forzar imagen a imagen",
|
"strength": "Forzar imagen a imagen",
|
||||||
"recallParameters": "Parámetros de recuperación",
|
"recallParameters": "Parámetros de recuperación",
|
||||||
"recallParameter": "Recuperar {{label}}",
|
|
||||||
"steps": "Pasos",
|
"steps": "Pasos",
|
||||||
"noRecallParameters": "Sin parámetros para recuperar",
|
"noRecallParameters": "Sin parámetros para recuperar"
|
||||||
"parsingFailed": "Error al analizar"
|
|
||||||
},
|
},
|
||||||
"system": {
|
"system": {
|
||||||
"logLevel": {
|
"logLevel": {
|
||||||
|
|||||||
@@ -28,7 +28,6 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "Kuvan koko",
|
"galleryImageSize": "Kuvan koko",
|
||||||
"gallerySettings": "Gallerian asetukset",
|
"gallerySettings": "Gallerian asetukset",
|
||||||
"autoSwitchNewImages": "Vaihda uusiin kuviin automaattisesti",
|
"autoSwitchNewImages": "Vaihda uusiin kuviin automaattisesti"
|
||||||
"noImagesInGallery": "Ei kuvia galleriassa"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,21 +27,15 @@
|
|||||||
"error": "Erreur",
|
"error": "Erreur",
|
||||||
"installed": "Installé",
|
"installed": "Installé",
|
||||||
"format": "format",
|
"format": "format",
|
||||||
"goTo": "Aller à",
|
|
||||||
"input": "Entrée",
|
"input": "Entrée",
|
||||||
"linear": "Linéaire",
|
"linear": "Linéaire",
|
||||||
"localSystem": "Système local",
|
|
||||||
"learnMore": "En savoir plus",
|
"learnMore": "En savoir plus",
|
||||||
"modelManager": "Gestionnaire de modèle",
|
"modelManager": "Gestionnaire de modèle",
|
||||||
"notInstalled": "Non $t(common.installed)",
|
|
||||||
"openInNewTab": "Ouvrir dans un nouvel onglet",
|
"openInNewTab": "Ouvrir dans un nouvel onglet",
|
||||||
"somethingWentWrong": "Une erreur s'est produite",
|
"somethingWentWrong": "Une erreur s'est produite",
|
||||||
"created": "Créé",
|
"created": "Créé",
|
||||||
"tab": "Onglet",
|
"tab": "Onglet",
|
||||||
"folder": "Dossier",
|
"folder": "Dossier",
|
||||||
"imageFailedToLoad": "Impossible de charger l'Image",
|
|
||||||
"prevPage": "Page précédente",
|
|
||||||
"nextPage": "Page suivante",
|
|
||||||
"selected": "Sélectionné",
|
"selected": "Sélectionné",
|
||||||
"save": "Enregistrer",
|
"save": "Enregistrer",
|
||||||
"updated": "Mis à jour",
|
"updated": "Mis à jour",
|
||||||
@@ -111,7 +105,6 @@
|
|||||||
"min": "Min",
|
"min": "Min",
|
||||||
"max": "Max",
|
"max": "Max",
|
||||||
"values": "Valeurs",
|
"values": "Valeurs",
|
||||||
"resetToDefaults": "Réinitialiser par défaut",
|
|
||||||
"seed": "Graine",
|
"seed": "Graine",
|
||||||
"combinatorial": "Combinatoire"
|
"combinatorial": "Combinatoire"
|
||||||
},
|
},
|
||||||
@@ -119,11 +112,9 @@
|
|||||||
"galleryImageSize": "Taille de l'image",
|
"galleryImageSize": "Taille de l'image",
|
||||||
"gallerySettings": "Paramètres de la galerie",
|
"gallerySettings": "Paramètres de la galerie",
|
||||||
"autoSwitchNewImages": "Basculer automatiquement vers de nouvelles images",
|
"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.",
|
"bulkDownloadRequestedDesc": "Votre demande de téléchargement est en cours de traitement. Cela peut prendre quelques instants.",
|
||||||
"deleteSelection": "Supprimer la sélection",
|
"deleteSelection": "Supprimer la sélection",
|
||||||
"selectAllOnPage": "Séléctionner toute la page",
|
"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.",
|
"featuresWillReset": "Si vous supprimez cette image, ces fonctionnalités vont être réinitialisés.",
|
||||||
"loading": "Chargement",
|
"loading": "Chargement",
|
||||||
"sortDirection": "Direction de tri",
|
"sortDirection": "Direction de tri",
|
||||||
@@ -149,7 +140,6 @@
|
|||||||
"openInViewer": "Ouvrir dans le Visualiseur",
|
"openInViewer": "Ouvrir dans le Visualiseur",
|
||||||
"showArchivedBoards": "Montrer les Planches archivées",
|
"showArchivedBoards": "Montrer les Planches archivées",
|
||||||
"selectForCompare": "Séléctionner pour comparaison",
|
"selectForCompare": "Séléctionner pour comparaison",
|
||||||
"selectAnImageToCompare": "Séléctionner une Image à comparer",
|
|
||||||
"exitCompare": "Sortir de la comparaison",
|
"exitCompare": "Sortir de la comparaison",
|
||||||
"compareHelp2": "Appuyez sur <Kbd>M</Kbd> pour faire défiler les modes de comparaison.",
|
"compareHelp2": "Appuyez sur <Kbd>M</Kbd> pour faire défiler les modes de comparaison.",
|
||||||
"swapImages": "Échanger les Images",
|
"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.",
|
"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.",
|
"compareHelp3": "Appuyer sur <Kbd>C</Kbd> pour échanger les images à comparer.",
|
||||||
"image": "image",
|
"image": "image",
|
||||||
"openViewer": "Ouvrir le Visualisateur",
|
|
||||||
"closeViewer": "Fermer le Visualisateur",
|
|
||||||
"currentlyInUse": "Cette image est actuellement utilisée dans ces fonctionalités :",
|
"currentlyInUse": "Cette image est actuellement utilisée dans ces fonctionalités :",
|
||||||
"jump": "Sauter",
|
|
||||||
"starImage": "Marquer l'Image",
|
"starImage": "Marquer l'Image",
|
||||||
"download": "Téléchargement",
|
"download": "Téléchargement",
|
||||||
"deleteImage_one": "Supprimer l'Image",
|
"deleteImage_one": "Supprimer l'Image",
|
||||||
@@ -247,7 +234,6 @@
|
|||||||
"metadata": "Métadonnées",
|
"metadata": "Métadonnées",
|
||||||
"scanFolder": "Scanner le dossier",
|
"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.",
|
"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",
|
"installQueue": "File d'attente d'installation",
|
||||||
"modelImageDeleteFailed": "Échec de la suppression de l'image du modèle",
|
"modelImageDeleteFailed": "Échec de la suppression de l'image du modèle",
|
||||||
"modelName": "Nom 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.",
|
"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",
|
"clipEmbed": "Intégration CLIP",
|
||||||
"spandrelImageToImage": "Image vers Image (Spandrel)",
|
"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",
|
"t5Encoder": "Encodeur T5",
|
||||||
"learnMoreAboutSupportedModels": "En savoir plus sur les modèles que nous prenons en charge",
|
"learnMoreAboutSupportedModels": "En savoir plus sur les modèles que nous prenons en charge",
|
||||||
"includesNModels": "Contient {{n}} modèles et leurs dépendances",
|
"includesNModels": "Contient {{n}} modèles et leurs dépendances",
|
||||||
@@ -346,12 +331,10 @@
|
|||||||
"infillMethod": "Méthode de Remplissage",
|
"infillMethod": "Méthode de Remplissage",
|
||||||
"tileSize": "Taille des Tuiles",
|
"tileSize": "Taille des Tuiles",
|
||||||
"copyImage": "Copier Image",
|
"copyImage": "Copier Image",
|
||||||
"downloadImage": "Télécharger Image",
|
|
||||||
"usePrompt": "Utiliser la suggestion",
|
"usePrompt": "Utiliser la suggestion",
|
||||||
"useSeed": "Utiliser la graine",
|
"useSeed": "Utiliser la graine",
|
||||||
"useAll": "Tout utiliser",
|
"useAll": "Tout utiliser",
|
||||||
"info": "Info",
|
"info": "Info",
|
||||||
"showOptionsPanel": "Afficher le panneau latéral (O ou T)",
|
|
||||||
"invoke": {
|
"invoke": {
|
||||||
"noPrompts": "Aucun prompts généré",
|
"noPrompts": "Aucun prompts généré",
|
||||||
"missingInputForField": "entrée manquante",
|
"missingInputForField": "entrée manquante",
|
||||||
@@ -362,21 +345,16 @@
|
|||||||
"noModelSelected": "Aucun modèle sélectionné",
|
"noModelSelected": "Aucun modèle sélectionné",
|
||||||
"noNodesInGraph": "Aucun nœud dans le graphique",
|
"noNodesInGraph": "Aucun nœud dans le graphique",
|
||||||
"systemDisconnected": "Système déconnecté",
|
"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",
|
"noFLUXVAEModelSelected": "Aucun modèle VAE sélectionné pour la génération FLUX",
|
||||||
"canvasIsTransforming": "La Toile est occupée (en transformation)",
|
"canvasIsTransforming": "La Toile est occupée (en transformation)",
|
||||||
"canvasIsRasterizing": "La Toile est occupée (en rastérisation)",
|
"canvasIsRasterizing": "La Toile est occupée (en rastérisation)",
|
||||||
"noCLIPEmbedModelSelected": "Aucun modèle CLIP Embed sélectionné pour la génération FLUX",
|
"noCLIPEmbedModelSelected": "Aucun modèle CLIP Embed sélectionné pour la génération FLUX",
|
||||||
"canvasIsFiltering": "La Toile est occupée (en filtration)",
|
"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",
|
"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)",
|
"canvasIsCompositing": "La Toile est occupée (en composition)",
|
||||||
"collectionTooFewItems": "trop peu d'éléments, minimum {{minItems}}",
|
"collectionTooFewItems": "trop peu d'éléments, minimum {{minItems}}",
|
||||||
"collectionTooManyItems": "trop d'éléments, maximum {{maxItems}}",
|
"collectionTooManyItems": "trop d'éléments, maximum {{maxItems}}",
|
||||||
"canvasIsSelectingObject": "La toile est occupée (sélection d'objet)",
|
"canvasIsSelectingObject": "La toile est occupée (sélection d'objet)",
|
||||||
"emptyBatches": "lots vides",
|
|
||||||
"batchNodeNotConnected": "Noeud de lots non connecté : {{label}}",
|
"batchNodeNotConnected": "Noeud de lots non connecté : {{label}}",
|
||||||
"fluxModelMultipleControlLoRAs": "Vous ne pouvez utiliser qu'un seul Control LoRA à la fois",
|
"fluxModelMultipleControlLoRAs": "Vous ne pouvez utiliser qu'un seul Control LoRA à la fois",
|
||||||
"collectionNumberLTMin": "{{value}} < {{minimum}} (incl. min)",
|
"collectionNumberLTMin": "{{value}} < {{minimum}} (incl. min)",
|
||||||
@@ -468,9 +446,7 @@
|
|||||||
"informationalPopoversDisabled": "Pop-ups d'information désactivés",
|
"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.",
|
"informationalPopoversDisabledDesc": "Les pop-ups d'information ont été désactivés. Activez-les dans les paramètres.",
|
||||||
"confirmOnNewSession": "Confirmer lors d'une nouvelle session",
|
"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",
|
"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"
|
"showDetailedInvocationProgress": "Afficher les détails de progression"
|
||||||
},
|
},
|
||||||
"toast": {
|
"toast": {
|
||||||
@@ -486,17 +462,14 @@
|
|||||||
"addedToBoard": "Ajouté aux ressources de la planche {{name}}",
|
"addedToBoard": "Ajouté aux ressources de la planche {{name}}",
|
||||||
"workflowLoaded": "Workflow chargé",
|
"workflowLoaded": "Workflow chargé",
|
||||||
"connected": "Connecté au serveur",
|
"connected": "Connecté au serveur",
|
||||||
"setNodeField": "Définir comme champ de nœud",
|
|
||||||
"imageUploadFailed": "Échec de l'importation de l'image",
|
"imageUploadFailed": "Échec de l'importation de l'image",
|
||||||
"loadedWithWarnings": "Workflow chargé avec des avertissements",
|
"loadedWithWarnings": "Workflow chargé avec des avertissements",
|
||||||
"imageUploaded": "Image importée",
|
"imageUploaded": "Image importée",
|
||||||
"modelAddedSimple": "Modèle ajouté à la file d'attente",
|
"modelAddedSimple": "Modèle ajouté à la file d'attente",
|
||||||
"setControlImage": "Définir comme image de contrôle",
|
|
||||||
"workflowDeleted": "Workflow supprimé",
|
"workflowDeleted": "Workflow supprimé",
|
||||||
"baseModelChangedCleared_one": "Effacé ou désactivé {{count}} sous-modèle incompatible",
|
"baseModelChangedCleared_one": "Effacé ou désactivé {{count}} sous-modèle incompatible",
|
||||||
"baseModelChangedCleared_many": "Effacé ou désactivé {{count}} sous-modèles incompatibles",
|
"baseModelChangedCleared_many": "Effacé ou désactivé {{count}} sous-modèles incompatibles",
|
||||||
"baseModelChangedCleared_other": "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",
|
"problemDownloadingImage": "Impossible de télécharger l'image",
|
||||||
"problemRetrievingWorkflow": "Problème de récupération du Workflow",
|
"problemRetrievingWorkflow": "Problème de récupération du Workflow",
|
||||||
"problemDeletingWorkflow": "Problème de suppression du Workflow",
|
"problemDeletingWorkflow": "Problème de suppression du Workflow",
|
||||||
@@ -510,12 +483,10 @@
|
|||||||
"errorCopied": "Erreur copiée",
|
"errorCopied": "Erreur copiée",
|
||||||
"parametersSet": "Paramètres rappelés",
|
"parametersSet": "Paramètres rappelés",
|
||||||
"somethingWentWrong": "Quelque chose a échoué",
|
"somethingWentWrong": "Quelque chose a échoué",
|
||||||
"imageSaved": "Image enregistrée",
|
|
||||||
"unableToLoadStylePreset": "Impossible de charger le préréglage de style",
|
"unableToLoadStylePreset": "Impossible de charger le préréglage de style",
|
||||||
"stylePresetLoaded": "Préréglage de style chargé",
|
"stylePresetLoaded": "Préréglage de style chargé",
|
||||||
"parameterNotSetDescWithMessage": "Impossible de rappeler {{parameter}} : {{message}}",
|
"parameterNotSetDescWithMessage": "Impossible de rappeler {{parameter}} : {{message}}",
|
||||||
"importFailed": "Importation échouée",
|
"importFailed": "Importation échouée",
|
||||||
"imageSavingFailed": "Échec de l'enregistrement de l'image",
|
|
||||||
"importSuccessful": "Importation réussie",
|
"importSuccessful": "Importation réussie",
|
||||||
"outOfMemoryError": "Erreur de mémoire insuffisante",
|
"outOfMemoryError": "Erreur de mémoire insuffisante",
|
||||||
"sessionRef": "Session : {{sessionId}}",
|
"sessionRef": "Session : {{sessionId}}",
|
||||||
@@ -523,16 +494,11 @@
|
|||||||
"parameterSetDesc": "Rappelé {{parameter}}",
|
"parameterSetDesc": "Rappelé {{parameter}}",
|
||||||
"parameterNotSetDesc": "Impossible de rappeler {{parameter}}",
|
"parameterNotSetDesc": "Impossible de rappeler {{parameter}}",
|
||||||
"layerCopiedToClipboard": "Calque copié dans le presse-papiers",
|
"layerCopiedToClipboard": "Calque copié dans le presse-papiers",
|
||||||
"layerSavedToAssets": "Calque enregistré dans les ressources",
|
|
||||||
"problemCopyingLayer": "Impossible de copier la couche",
|
"problemCopyingLayer": "Impossible de copier la couche",
|
||||||
"baseModelChanged": "Modèle de base changé",
|
"baseModelChanged": "Modèle de base changé",
|
||||||
"problemSavingLayer": "Impossible d'enregistrer la couche",
|
"problemSavingLayer": "Impossible d'enregistrer la couche",
|
||||||
"imageNotLoadedDesc": "Image introuvable",
|
|
||||||
"linkCopied": "Lien copié",
|
"linkCopied": "Lien copié",
|
||||||
"imagesWillBeAddedTo": "Les images Importées seront ajoutées au ressources de la Planche {{boardName}}.",
|
"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)",
|
"addedToUncategorized": "Ajouté aux ressources de la planche $t(boards.uncategorized)",
|
||||||
"pasteSuccess": "Collé à {{destination}}",
|
"pasteSuccess": "Collé à {{destination}}",
|
||||||
"pasteFailed": "Échec du collage",
|
"pasteFailed": "Échec du collage",
|
||||||
@@ -580,8 +546,6 @@
|
|||||||
"movingImagesToBoard_one": "Déplacer {{count}} image à cette planche :",
|
"movingImagesToBoard_one": "Déplacer {{count}} image à cette planche :",
|
||||||
"movingImagesToBoard_many": "Déplacer {{count}} images à cette planche :",
|
"movingImagesToBoard_many": "Déplacer {{count}} images à cette planche :",
|
||||||
"movingImagesToBoard_other": "Déplacer {{count}} image à cette planche :",
|
"movingImagesToBoard_other": "Déplacer {{count}} image à cette planche :",
|
||||||
"viewBoards": "Voir les Planches",
|
|
||||||
"hideBoards": "Cacher les Planches",
|
|
||||||
"noBoards": "Pas de Planches {{boardType}}",
|
"noBoards": "Pas de Planches {{boardType}}",
|
||||||
"shared": "Planches Partagées",
|
"shared": "Planches Partagées",
|
||||||
"searchBoard": "Chercher les Planches...",
|
"searchBoard": "Chercher les Planches...",
|
||||||
@@ -681,7 +645,6 @@
|
|||||||
"batchQueued": "Lot ajouté à la file d'attente",
|
"batchQueued": "Lot ajouté à la file d'attente",
|
||||||
"gallery": "Galerie",
|
"gallery": "Galerie",
|
||||||
"notReady": "Impossible d'ajouter à la file d'attente",
|
"notReady": "Impossible d'ajouter à la file d'attente",
|
||||||
"batchFieldValues": "Valeurs Champ Lot",
|
|
||||||
"front": "début",
|
"front": "début",
|
||||||
"graphQueued": "Graph ajouté à la file d'attente",
|
"graphQueued": "Graph ajouté à la file d'attente",
|
||||||
"other": "Autre",
|
"other": "Autre",
|
||||||
@@ -712,13 +675,11 @@
|
|||||||
"compatibleEmbeddings": "Embeddings Compatibles"
|
"compatibleEmbeddings": "Embeddings Compatibles"
|
||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"upscaleMethod": "Méthode d'Agrandissement",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"enabled": "Correction Haute Résolution Activée",
|
"enabled": "Correction Haute Résolution Activée",
|
||||||
"strength": "Force de la Correction Haute Résolution",
|
"strength": "Force de la Correction Haute Résolution",
|
||||||
"method": "Méthode 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"
|
"hrf": "Correction Haute Résolution"
|
||||||
},
|
},
|
||||||
"invocationCache": {
|
"invocationCache": {
|
||||||
@@ -901,10 +862,6 @@
|
|||||||
"desc": "Définit le zoom de la toile à 400 %.",
|
"desc": "Définit le zoom de la toile à 400 %.",
|
||||||
"title": "Zoomer à 400 %"
|
"title": "Zoomer à 400 %"
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "Définir la couleur sur blanc",
|
|
||||||
"desc": "Définir la couleur de l'outil actuel sur blanc."
|
|
||||||
},
|
|
||||||
"transformSelected": {
|
"transformSelected": {
|
||||||
"title": "Transformer",
|
"title": "Transformer",
|
||||||
"desc": "Transforme la couche sélectionnée."
|
"desc": "Transforme la couche sélectionnée."
|
||||||
@@ -1490,8 +1447,7 @@
|
|||||||
"showDynamicPrompts": "Afficher les Prompts dynamiques",
|
"showDynamicPrompts": "Afficher les Prompts dynamiques",
|
||||||
"dynamicPrompts": "Prompts Dynamiques",
|
"dynamicPrompts": "Prompts Dynamiques",
|
||||||
"promptsPreview": "Prévisualisation des Prompts",
|
"promptsPreview": "Prévisualisation des Prompts",
|
||||||
"loading": "Génération des Pompts Dynamiques...",
|
"loading": "Génération des Pompts Dynamiques..."
|
||||||
"promptsToGenerate": "Prompts à générer"
|
|
||||||
},
|
},
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"positivePrompt": "Prompt Positif",
|
"positivePrompt": "Prompt Positif",
|
||||||
@@ -1519,18 +1475,12 @@
|
|||||||
"recallParameters": "Rappeler les paramètres",
|
"recallParameters": "Rappeler les paramètres",
|
||||||
"imageDimensions": "Dimensions de l'image",
|
"imageDimensions": "Dimensions de l'image",
|
||||||
"parameterSet": "Paramètre {{parameter}} défini",
|
"parameterSet": "Paramètre {{parameter}} défini",
|
||||||
"parsingFailed": "L'analyse a échoué",
|
|
||||||
"recallParameter": "Rappeler {{label}}",
|
|
||||||
"canvasV2Metadata": "Toile",
|
"canvasV2Metadata": "Toile",
|
||||||
"guidance": "Guide",
|
"guidance": "Guide",
|
||||||
"seamlessXAxis": "Axe X sans bords",
|
"seamlessXAxis": "Axe X sans bords",
|
||||||
"seamlessYAxis": "Axe Y sans bords"
|
"seamlessYAxis": "Axe Y sans bords"
|
||||||
},
|
},
|
||||||
"sdxl": {
|
"sdxl": {
|
||||||
"freePromptStyle": "Écriture de Prompt manuelle",
|
|
||||||
"concatPromptStyle": "Lier Prompt & Style",
|
|
||||||
"negStylePrompt": "Style Prompt Négatif",
|
|
||||||
"posStylePrompt": "Style Prompt Positif",
|
|
||||||
"refinerStart": "Démarrer le Refiner",
|
"refinerStart": "Démarrer le Refiner",
|
||||||
"denoisingStrength": "Force de débruitage",
|
"denoisingStrength": "Force de débruitage",
|
||||||
"steps": "Étapes",
|
"steps": "Étapes",
|
||||||
@@ -1547,8 +1497,6 @@
|
|||||||
"nodes": {
|
"nodes": {
|
||||||
"showMinimapnodes": "Afficher la MiniCarte",
|
"showMinimapnodes": "Afficher la MiniCarte",
|
||||||
"fitViewportNodes": "Ajuster la Vue",
|
"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",
|
"hideMinimapnodes": "Masquer MiniCarte",
|
||||||
"zoomOutNodes": "Dézoomer",
|
"zoomOutNodes": "Dézoomer",
|
||||||
"zoomInNodes": "Zoomer",
|
"zoomInNodes": "Zoomer",
|
||||||
@@ -1572,9 +1520,7 @@
|
|||||||
"colorCodeEdges": "Code de couleur des connexions",
|
"colorCodeEdges": "Code de couleur des connexions",
|
||||||
"colorCodeEdgesHelp": "Code couleur des connexions en fonction de leurs champs connectés",
|
"colorCodeEdgesHelp": "Code couleur des connexions en fonction de leurs champs connectés",
|
||||||
"currentImage": "Image actuelle",
|
"currentImage": "Image actuelle",
|
||||||
"noFieldsLinearview": "Aucun champ ajouté à la vue linéaire",
|
|
||||||
"float": "Flottant",
|
"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é ?)",
|
"missingTemplate": "Nœud invalide : le nœud {{node}} de type {{type}} modèle manquant (non installé ?)",
|
||||||
"noWorkflow": "Pas de Workflow",
|
"noWorkflow": "Pas de Workflow",
|
||||||
"validateConnectionsHelp": "Prévenir la création de connexions invalides et l'invocation de graphes invalides",
|
"validateConnectionsHelp": "Prévenir la création de connexions invalides et l'invocation de graphes invalides",
|
||||||
@@ -1585,12 +1531,10 @@
|
|||||||
"scheduler": "Planificateur",
|
"scheduler": "Planificateur",
|
||||||
"notes": "Notes",
|
"notes": "Notes",
|
||||||
"notesDescription": "Ajouter des notes sur votre workflow",
|
"notesDescription": "Ajouter des notes sur votre workflow",
|
||||||
"unableToLoadWorkflow": "Impossible de charger le Workflow",
|
|
||||||
"addNode": "Ajouter un nœud",
|
"addNode": "Ajouter un nœud",
|
||||||
"problemSettingTitle": "Problème lors de définition du Titre",
|
"problemSettingTitle": "Problème lors de définition du Titre",
|
||||||
"connectionWouldCreateCycle": "La connexion créerait un cycle",
|
"connectionWouldCreateCycle": "La connexion créerait un cycle",
|
||||||
"currentImageDescription": "Affiche l'image actuelle dans l'éditeur de nœuds",
|
"currentImageDescription": "Affiche l'image actuelle dans l'éditeur de nœuds",
|
||||||
"versionUnknown": " Version inconnue",
|
|
||||||
"cannotConnectInputToInput": "Impossible de connecter l'entrée à l'entrée",
|
"cannotConnectInputToInput": "Impossible de connecter l'entrée à l'entrée",
|
||||||
"addNodeToolTip": "Ajouter un nœud (Shift+A, Espace)",
|
"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",
|
"fullyContainNodesHelp": "Les nœuds doivent être entièrement à l'intérieur de la zone de sélection pour être sélectionnés",
|
||||||
@@ -1606,7 +1550,6 @@
|
|||||||
"nodeSearch": "Rechercher des nœuds",
|
"nodeSearch": "Rechercher des nœuds",
|
||||||
"collection": "Collection",
|
"collection": "Collection",
|
||||||
"noOutputRecorded": "Aucun résultat enregistré",
|
"noOutputRecorded": "Aucun résultat enregistré",
|
||||||
"removeLinearView": "Retirer de la vue linéaire",
|
|
||||||
"snapToGrid": "Aligner sur la grille",
|
"snapToGrid": "Aligner sur la grille",
|
||||||
"workflow": "Workflow",
|
"workflow": "Workflow",
|
||||||
"updateApp": "Mettre à jour l'application",
|
"updateApp": "Mettre à jour l'application",
|
||||||
@@ -1615,7 +1558,6 @@
|
|||||||
"noConnectionInProgress": "Aucune connexion en cours",
|
"noConnectionInProgress": "Aucune connexion en cours",
|
||||||
"nodeType": "Type de nœud",
|
"nodeType": "Type de nœud",
|
||||||
"workflowContact": "Contact",
|
"workflowContact": "Contact",
|
||||||
"unknownTemplate": "Modèle inconnu",
|
|
||||||
"unknownNode": "Nœud inconnu",
|
"unknownNode": "Nœud inconnu",
|
||||||
"workflowVersion": "Version",
|
"workflowVersion": "Version",
|
||||||
"string": "Chaîne de caractères",
|
"string": "Chaîne de caractères",
|
||||||
@@ -1629,7 +1571,6 @@
|
|||||||
"cannotDuplicateConnection": "Impossible de créer des connexions en double",
|
"cannotDuplicateConnection": "Impossible de créer des connexions en double",
|
||||||
"resetToDefaultValue": "Réinitialiser à la valeur par défaut",
|
"resetToDefaultValue": "Réinitialiser à la valeur par défaut",
|
||||||
"unknownNodeType": "Type de nœud inconnu",
|
"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.",
|
"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",
|
"nodePack": "Paquet de nœuds",
|
||||||
"sourceNodeDoesNotExist": "Connexion invalide : le nœud source/de sortie {{node}} n'existe pas",
|
"sourceNodeDoesNotExist": "Connexion invalide : le nœud source/de sortie {{node}} n'existe pas",
|
||||||
@@ -1644,7 +1585,6 @@
|
|||||||
"clearWorkflow": "Effacer le Workflow",
|
"clearWorkflow": "Effacer le Workflow",
|
||||||
"clearWorkflowDesc": "Effacer ce workflow et en commencer un nouveau ?",
|
"clearWorkflowDesc": "Effacer ce workflow et en commencer un nouveau ?",
|
||||||
"unsupportedArrayItemType": "type d'élément de tableau non pris en charge \"{{type}}\"",
|
"unsupportedArrayItemType": "type d'élément de tableau non pris en charge \"{{type}}\"",
|
||||||
"addLinearView": "Ajouter à la vue linéaire",
|
|
||||||
"collectionOrScalarFieldType": "{{name}} (Unique ou Collection)",
|
"collectionOrScalarFieldType": "{{name}} (Unique ou Collection)",
|
||||||
"unableToExtractEnumOptions": "impossible d'extraire les options d'énumération",
|
"unableToExtractEnumOptions": "impossible d'extraire les options d'énumération",
|
||||||
"unsupportedAnyOfLength": "trop de membres dans l'union ({{count}})",
|
"unsupportedAnyOfLength": "trop de membres dans l'union ({{count}})",
|
||||||
@@ -1652,7 +1592,6 @@
|
|||||||
"viewMode": "Utiliser en vue linéaire",
|
"viewMode": "Utiliser en vue linéaire",
|
||||||
"collectionFieldType": "{{name}} (Collection)",
|
"collectionFieldType": "{{name}} (Collection)",
|
||||||
"newWorkflow": "Nouveau Workflow",
|
"newWorkflow": "Nouveau Workflow",
|
||||||
"reorderLinearView": "Réorganiser la vue linéaire",
|
|
||||||
"outputFieldTypeParseError": "Impossible d'analyser le type du champ de sortie {{node}}.{{field}} ({{message}})",
|
"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}}",
|
"unsupportedMismatchedUnion": "type CollectionOrScalar non concordant avec les types de base {{firstType}} et {{secondType}}",
|
||||||
"unableToParseFieldType": "impossible d'analyser le type de champ",
|
"unableToParseFieldType": "impossible d'analyser le type de champ",
|
||||||
@@ -1686,13 +1625,9 @@
|
|||||||
"arithmeticSequence": "Séquence Arithmétique",
|
"arithmeticSequence": "Séquence Arithmétique",
|
||||||
"uniformRandomDistribution": "Distribution Aléatoire Uniforme",
|
"uniformRandomDistribution": "Distribution Aléatoire Uniforme",
|
||||||
"noBatchGroup": "aucun groupe",
|
"noBatchGroup": "aucun groupe",
|
||||||
"generatorLoading": "chargement",
|
|
||||||
"generatorLoadFromFile": "Charger depuis un Fichier",
|
"generatorLoadFromFile": "Charger depuis un Fichier",
|
||||||
"dynamicPromptsRandom": "Prompts Dynamiques (Aléatoire)",
|
"dynamicPromptsRandom": "Prompts Dynamiques (Aléatoire)",
|
||||||
"integerRangeGenerator": "Générateur d'interval d'entiers",
|
|
||||||
"generateValues": "Générer Valeurs",
|
|
||||||
"linearDistribution": "Distribution Linéaire",
|
"linearDistribution": "Distribution Linéaire",
|
||||||
"floatRangeGenerator": "Générateur d'interval de nombres décimaux",
|
|
||||||
"generatorNRandomValues_one": "{{count}} valeur aléatoire",
|
"generatorNRandomValues_one": "{{count}} valeur aléatoire",
|
||||||
"generatorNRandomValues_many": "{{count}} valeurs aléatoires",
|
"generatorNRandomValues_many": "{{count}} valeurs aléatoires",
|
||||||
"generatorNRandomValues_other": "{{count}} valeurs aléatoires",
|
"generatorNRandomValues_other": "{{count}} valeurs aléatoires",
|
||||||
@@ -1712,7 +1647,6 @@
|
|||||||
"generatorImagesCategory": "Catégorie",
|
"generatorImagesCategory": "Catégorie",
|
||||||
"generatorImagesFromBoard": "Images de la Planche",
|
"generatorImagesFromBoard": "Images de la Planche",
|
||||||
"missingSourceOrTargetHandle": "Manque de gestionnaire source ou cible",
|
"missingSourceOrTargetHandle": "Manque de gestionnaire source ou cible",
|
||||||
"loadingTemplates": "Chargement de {{name}}",
|
|
||||||
"loadWorkflowDesc2": "Votre workflow actuel contient des modifications non enregistrées.",
|
"loadWorkflowDesc2": "Votre workflow actuel contient des modifications non enregistrées.",
|
||||||
"generatorImages_one": "{{count}} image",
|
"generatorImages_one": "{{count}} image",
|
||||||
"generatorImages_many": "{{count}} images",
|
"generatorImages_many": "{{count}} images",
|
||||||
@@ -1723,10 +1657,8 @@
|
|||||||
"noModelsAvailable": "Aucun modèle disponible",
|
"noModelsAvailable": "Aucun modèle disponible",
|
||||||
"loading": "chargement",
|
"loading": "chargement",
|
||||||
"selectModel": "Sélectionner un modèle",
|
"selectModel": "Sélectionner un modèle",
|
||||||
"noMatchingLoRAs": "Aucun LoRA correspondant",
|
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
"noRefinerModelsInstalled": "Aucun modèle SDXL Refiner installé",
|
"noRefinerModelsInstalled": "Aucun modèle SDXL Refiner installé",
|
||||||
"noLoRAsInstalled": "Aucun LoRA installé",
|
|
||||||
"addLora": "Ajouter LoRA",
|
"addLora": "Ajouter LoRA",
|
||||||
"defaultVAE": "VAE par défaut",
|
"defaultVAE": "VAE par défaut",
|
||||||
"concepts": "Concepts"
|
"concepts": "Concepts"
|
||||||
@@ -1734,11 +1666,8 @@
|
|||||||
"workflows": {
|
"workflows": {
|
||||||
"workflowLibrary": "Bibliothèque",
|
"workflowLibrary": "Bibliothèque",
|
||||||
"loading": "Chargement des Workflows",
|
"loading": "Chargement des Workflows",
|
||||||
"searchWorkflows": "Chercher des Workflows",
|
|
||||||
"workflowCleared": "Workflow effacé",
|
"workflowCleared": "Workflow effacé",
|
||||||
"noDescription": "Aucune description",
|
|
||||||
"deleteWorkflow": "Supprimer le Workflow",
|
"deleteWorkflow": "Supprimer le Workflow",
|
||||||
"openWorkflow": "Ouvrir le Workflow",
|
|
||||||
"uploadWorkflow": "Charger à partir d'un fichier",
|
"uploadWorkflow": "Charger à partir d'un fichier",
|
||||||
"workflowName": "Nom du Workflow",
|
"workflowName": "Nom du Workflow",
|
||||||
"unnamedWorkflow": "Workflow sans nom",
|
"unnamedWorkflow": "Workflow sans nom",
|
||||||
@@ -1751,8 +1680,6 @@
|
|||||||
"problemSavingWorkflow": "Problème de sauvegarde du Workflow",
|
"problemSavingWorkflow": "Problème de sauvegarde du Workflow",
|
||||||
"workflowEditorMenu": "Menu de l'Éditeur de Workflow",
|
"workflowEditorMenu": "Menu de l'Éditeur de Workflow",
|
||||||
"newWorkflowCreated": "Nouveau Workflow créé",
|
"newWorkflowCreated": "Nouveau Workflow créé",
|
||||||
"clearWorkflowSearchFilter": "Réinitialiser le filtre de recherche de Workflow",
|
|
||||||
"problemLoading": "Problème de chargement des Workflows",
|
|
||||||
"workflowSaved": "Workflow enregistré",
|
"workflowSaved": "Workflow enregistré",
|
||||||
"noWorkflows": "Pas de Workflows",
|
"noWorkflows": "Pas de Workflows",
|
||||||
"ascending": "Ascendant",
|
"ascending": "Ascendant",
|
||||||
@@ -1765,9 +1692,6 @@
|
|||||||
"opened": "Ouvert",
|
"opened": "Ouvert",
|
||||||
"name": "Nom",
|
"name": "Nom",
|
||||||
"autoLayout": "Mise en page automatique",
|
"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",
|
"copyShareLink": "Copier le lien de partage",
|
||||||
"chooseWorkflowFromLibrary": "Choisir le Workflow dans la Bibliothèque",
|
"chooseWorkflowFromLibrary": "Choisir le Workflow dans la Bibliothèque",
|
||||||
"edit": "Modifer",
|
"edit": "Modifer",
|
||||||
@@ -1784,7 +1708,6 @@
|
|||||||
"multiLine": "Multi Ligne",
|
"multiLine": "Multi Ligne",
|
||||||
"headingPlaceholder": "En-tête vide",
|
"headingPlaceholder": "En-tête vide",
|
||||||
"emptyRootPlaceholderEditMode": "Faites glisser un élément de formulaire ou un champ de nœud ici pour commencer.",
|
"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",
|
"containerPlaceholder": "Conteneur Vide",
|
||||||
"row": "Ligne",
|
"row": "Ligne",
|
||||||
"column": "Colonne",
|
"column": "Colonne",
|
||||||
@@ -1798,10 +1721,8 @@
|
|||||||
"builder": "Constructeur de Formulaire",
|
"builder": "Constructeur de Formulaire",
|
||||||
"resetAllNodeFields": "Réinitialiser tous les champs de nœud",
|
"resetAllNodeFields": "Réinitialiser tous les champs de nœud",
|
||||||
"deleteAllElements": "Supprimer tous les éléments de formulaire",
|
"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"
|
"showDescription": "Afficher la description"
|
||||||
},
|
}
|
||||||
"openLibrary": "Ouvrir la Bibliothèque"
|
|
||||||
},
|
},
|
||||||
"whatsNew": {
|
"whatsNew": {
|
||||||
"whatsNewInInvoke": "Quoi de neuf dans Invoke",
|
"whatsNewInInvoke": "Quoi de neuf dans Invoke",
|
||||||
@@ -1810,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.",
|
"<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."
|
"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",
|
"readReleaseNotes": "Notes de version"
|
||||||
"watchUiUpdatesOverview": "Aperçu des mises à jour de l'interface utilisateur"
|
|
||||||
},
|
},
|
||||||
"ui": {
|
"ui": {
|
||||||
"tabs": {
|
"tabs": {
|
||||||
@@ -1828,7 +1748,6 @@
|
|||||||
},
|
},
|
||||||
"controlLayers": {
|
"controlLayers": {
|
||||||
"newLayerFromImage": "Nouvelle couche à partir de l'image",
|
"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",
|
"sendToCanvas": "Envoyer vers la Toile",
|
||||||
"globalReferenceImage": "Image de référence globale",
|
"globalReferenceImage": "Image de référence globale",
|
||||||
"newCanvasFromImage": "Nouvelle Toile à partir de l'image",
|
"newCanvasFromImage": "Nouvelle Toile à partir de l'image",
|
||||||
@@ -1984,7 +1903,6 @@
|
|||||||
},
|
},
|
||||||
"bookmark": "Marque-page pour Changement Rapide",
|
"bookmark": "Marque-page pour Changement Rapide",
|
||||||
"saveLayerToAssets": "Enregistrer la couche dans les ressources",
|
"saveLayerToAssets": "Enregistrer la couche dans les ressources",
|
||||||
"stagingOnCanvas": "Mise en attente des images sur",
|
|
||||||
"enableTransparencyEffect": "Activer l'effet de transparence",
|
"enableTransparencyEffect": "Activer l'effet de transparence",
|
||||||
"hidingType": "Masquer {{type}}",
|
"hidingType": "Masquer {{type}}",
|
||||||
"settings": {
|
"settings": {
|
||||||
@@ -2015,11 +1933,6 @@
|
|||||||
"disableAutoNegative": "Désactiver l'Auto Négatif",
|
"disableAutoNegative": "Désactiver l'Auto Négatif",
|
||||||
"addNegativePrompt": "Ajouter $t(controlLayers.negativePrompt)",
|
"addNegativePrompt": "Ajouter $t(controlLayers.negativePrompt)",
|
||||||
"addRegionalGuidance": "Ajouter $t(controlLayers.regionalGuidance)",
|
"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_one": "Couche",
|
||||||
"layer_many": "Couches",
|
"layer_many": "Couches",
|
||||||
"layer_other": "Couches",
|
"layer_other": "Couches",
|
||||||
@@ -2069,8 +1982,6 @@
|
|||||||
"next": "Suivant",
|
"next": "Suivant",
|
||||||
"saveToGallery": "Enregistrer dans la galerie"
|
"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",
|
"mergeVisibleError": "Erreur lors de la fusion des calques visibles",
|
||||||
"mergeVisibleOk": "Couches visibles fusionnées",
|
"mergeVisibleOk": "Couches visibles fusionnées",
|
||||||
"clearHistory": "Effacer l'historique",
|
"clearHistory": "Effacer l'historique",
|
||||||
@@ -2079,8 +1990,6 @@
|
|||||||
"duplicate": "Dupliquer",
|
"duplicate": "Dupliquer",
|
||||||
"enableAutoNegative": "Activer l'Auto Négatif",
|
"enableAutoNegative": "Activer l'Auto Négatif",
|
||||||
"showHUD": "Afficher HUD",
|
"showHUD": "Afficher HUD",
|
||||||
"sendToGallery": "Envoyer à la galerie",
|
|
||||||
"sendingToGallery": "Envoi des générations à la galerie",
|
|
||||||
"disableTransparencyEffect": "Désactiver l'effet de transparence",
|
"disableTransparencyEffect": "Désactiver l'effet de transparence",
|
||||||
"HUD": {
|
"HUD": {
|
||||||
"entityStatus": {
|
"entityStatus": {
|
||||||
@@ -2097,16 +2006,11 @@
|
|||||||
"opacity": "Opacité",
|
"opacity": "Opacité",
|
||||||
"savedToGalleryError": "Erreur lors de l'enregistrement dans la galerie",
|
"savedToGalleryError": "Erreur lors de l'enregistrement dans la galerie",
|
||||||
"addInpaintMask": "Ajouter $t(controlLayers.inpaintMask)",
|
"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",
|
"canvas": "Toile",
|
||||||
"savedToGalleryOk": "Enregistré dans la galerie",
|
"savedToGalleryOk": "Enregistré dans la galerie",
|
||||||
"addPositivePrompt": "Ajouter $t(controlLayers.prompt)",
|
"addPositivePrompt": "Ajouter $t(controlLayers.prompt)",
|
||||||
"showProgressOnCanvas": "Afficher la progression sur la Toile",
|
"showProgressOnCanvas": "Afficher la progression sur la Toile",
|
||||||
"newGallerySession": "Nouvelle session de galerie",
|
|
||||||
"newCanvasSession": "Nouvelle session de toile",
|
|
||||||
"showingType": "Afficher {{type}}",
|
"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)",
|
"addControlLayer": "Ajouter $t(controlLayers.controlLayer)",
|
||||||
"global": "Global",
|
"global": "Global",
|
||||||
"newGlobalReferenceImageOk": "Image de référence globale créée",
|
"newGlobalReferenceImageOk": "Image de référence globale créée",
|
||||||
@@ -2120,16 +2024,6 @@
|
|||||||
"newRasterLayerError": "Problème de création de couche de rastérisation",
|
"newRasterLayerError": "Problème de création de couche de rastérisation",
|
||||||
"negativePrompt": "Prompt négatif",
|
"negativePrompt": "Prompt négatif",
|
||||||
"weight": "Poids",
|
"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": {
|
||||||
"controlMode": "Mode de contrôle",
|
"controlMode": "Mode de contrôle",
|
||||||
"balanced": "Équilibré",
|
"balanced": "Équilibré",
|
||||||
@@ -2153,18 +2047,14 @@
|
|||||||
},
|
},
|
||||||
"fitBboxToLayers": "Ajuster la bounding box aux calques",
|
"fitBboxToLayers": "Ajuster la bounding box aux calques",
|
||||||
"regionIsEmpty": "La zone sélectionnée est vide",
|
"regionIsEmpty": "La zone sélectionnée est vide",
|
||||||
"controlLayers_withCount_visible": "Couches de contrôle ({{count}})",
|
|
||||||
"cropLayerToBbox": "Rogner la couche selon la bounding box",
|
"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",
|
"copyToClipboard": "Copier dans le presse-papiers",
|
||||||
"regionalGuidance_withCount_one": "$t(controlLayers.regionalGuidance)",
|
"regionalGuidance_withCount_one": "$t(controlLayers.regionalGuidance)",
|
||||||
"regionalGuidance_withCount_many": "Guidage Régional",
|
"regionalGuidance_withCount_many": "Guidage Régional",
|
||||||
"regionalGuidance_withCount_other": "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_one": "$t(controlLayers.inpaintMask)",
|
||||||
"inpaintMask_withCount_many": "Remplir les masques",
|
"inpaintMask_withCount_many": "Remplir les masques",
|
||||||
"inpaintMask_withCount_other": "Remplir les masques",
|
"inpaintMask_withCount_other": "Remplir les masques",
|
||||||
"newImg2ImgCanvasFromImage": "Nouvelle Img2Img à partir de l'image",
|
|
||||||
"bboxOverlay": "Afficher la superposition des Bounding Box",
|
"bboxOverlay": "Afficher la superposition des Bounding Box",
|
||||||
"moveToFront": "Déplacer vers le permier plan",
|
"moveToFront": "Déplacer vers le permier plan",
|
||||||
"moveToBack": "Déplacer vers l'arrière plan",
|
"moveToBack": "Déplacer vers l'arrière plan",
|
||||||
@@ -2179,7 +2069,6 @@
|
|||||||
"inpaintMask": "Masque de remplissage",
|
"inpaintMask": "Masque de remplissage",
|
||||||
"deleteReferenceImage": "Supprimer l'image de référence",
|
"deleteReferenceImage": "Supprimer l'image de référence",
|
||||||
"addReferenceImage": "Ajouter $t(controlLayers.referenceImage)",
|
"addReferenceImage": "Ajouter $t(controlLayers.referenceImage)",
|
||||||
"addGlobalReferenceImage": "Ajouter $t(controlLayers.globalReferenceImage)",
|
|
||||||
"removeBookmark": "Supprimer le marque-page",
|
"removeBookmark": "Supprimer le marque-page",
|
||||||
"regionalGuidance": "Guide régional",
|
"regionalGuidance": "Guide régional",
|
||||||
"regionalReferenceImage": "Image de référence régionale",
|
"regionalReferenceImage": "Image de référence régionale",
|
||||||
@@ -2208,16 +2097,12 @@
|
|||||||
"pointType": "Type de point",
|
"pointType": "Type de point",
|
||||||
"exclude": "Exclure",
|
"exclude": "Exclure",
|
||||||
"process": "Traiter",
|
"process": "Traiter",
|
||||||
"reset": "Réinitialiser",
|
"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."
|
|
||||||
},
|
},
|
||||||
"convertRegionalGuidanceTo": "Convertir $t(controlLayers.regionalGuidance) vers",
|
"convertRegionalGuidanceTo": "Convertir $t(controlLayers.regionalGuidance) vers",
|
||||||
"copyRasterLayerTo": "Copier $t(controlLayers.rasterLayer) vers",
|
"copyRasterLayerTo": "Copier $t(controlLayers.rasterLayer) vers",
|
||||||
"newControlLayer": "Nouveau $t(controlLayers.controlLayer)",
|
"newControlLayer": "Nouveau $t(controlLayers.controlLayer)",
|
||||||
"newRegionalGuidance": "Nouveau $t(controlLayers.regionalGuidance)",
|
"newRegionalGuidance": "Nouveau $t(controlLayers.regionalGuidance)",
|
||||||
"replaceCurrent": "Remplacer Actuel",
|
|
||||||
"convertControlLayerTo": "Convertir $t(controlLayers.controlLayer) vers",
|
"convertControlLayerTo": "Convertir $t(controlLayers.controlLayer) vers",
|
||||||
"convertInpaintMaskTo": "Convertir $t(controlLayers.inpaintMask) vers",
|
"convertInpaintMaskTo": "Convertir $t(controlLayers.inpaintMask) vers",
|
||||||
"copyControlLayerTo": "Copier $t(controlLayers.controlLayer) vers",
|
"copyControlLayerTo": "Copier $t(controlLayers.controlLayer) vers",
|
||||||
@@ -2263,9 +2148,7 @@
|
|||||||
"pasteToBboxDesc": "Nouvelle couche (dans Bbox)",
|
"pasteToBboxDesc": "Nouvelle couche (dans Bbox)",
|
||||||
"pasteToCanvasDesc": "Nouvelle couche (dans la Toile)",
|
"pasteToCanvasDesc": "Nouvelle couche (dans la Toile)",
|
||||||
"useImage": "Utiliser l'image",
|
"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."
|
||||||
"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)"
|
|
||||||
},
|
},
|
||||||
"upscaling": {
|
"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.",
|
"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": {
|
"gallery": {
|
||||||
"galleryImageSize": "גודל תמונה",
|
"galleryImageSize": "גודל תמונה",
|
||||||
"gallerySettings": "הגדרות גלריה",
|
"gallerySettings": "הגדרות גלריה",
|
||||||
"autoSwitchNewImages": "החלף אוטומטית לתמונות חדשות",
|
"autoSwitchNewImages": "החלף אוטומטית לתמונות חדשות"
|
||||||
"noImagesInGallery": "אין תמונות בגלריה"
|
|
||||||
},
|
},
|
||||||
"parameters": {
|
"parameters": {
|
||||||
"images": "תמונות",
|
"images": "תמונות",
|
||||||
@@ -70,12 +69,10 @@
|
|||||||
"tileSize": "גודל אריח",
|
"tileSize": "גודל אריח",
|
||||||
"symmetry": "סימטריה",
|
"symmetry": "סימטריה",
|
||||||
"copyImage": "העתקת תמונה",
|
"copyImage": "העתקת תמונה",
|
||||||
"downloadImage": "הורדת תמונה",
|
|
||||||
"usePrompt": "שימוש בבקשה",
|
"usePrompt": "שימוש בבקשה",
|
||||||
"useSeed": "שימוש בזרע",
|
"useSeed": "שימוש בזרע",
|
||||||
"useAll": "שימוש בהכל",
|
"useAll": "שימוש בהכל",
|
||||||
"info": "פרטים",
|
"info": "פרטים",
|
||||||
"showOptionsPanel": "הצג חלונית אפשרויות",
|
|
||||||
"shuffle": "ערבוב",
|
"shuffle": "ערבוב",
|
||||||
"noiseThreshold": "סף רעש",
|
"noiseThreshold": "סף רעש",
|
||||||
"perlinNoise": "רעש פרלין",
|
"perlinNoise": "רעש פרלין",
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -27,7 +27,6 @@
|
|||||||
"openInNewTab": "新しいタブで開く",
|
"openInNewTab": "新しいタブで開く",
|
||||||
"controlNet": "コントロールネット",
|
"controlNet": "コントロールネット",
|
||||||
"linear": "リニア",
|
"linear": "リニア",
|
||||||
"imageFailedToLoad": "画像が読み込めません",
|
|
||||||
"modelManager": "モデルマネージャー",
|
"modelManager": "モデルマネージャー",
|
||||||
"learnMore": "もっと学ぶ",
|
"learnMore": "もっと学ぶ",
|
||||||
"random": "ランダム",
|
"random": "ランダム",
|
||||||
@@ -56,7 +55,6 @@
|
|||||||
"details": "詳細",
|
"details": "詳細",
|
||||||
"inpaint": "inpaint",
|
"inpaint": "inpaint",
|
||||||
"delete": "削除",
|
"delete": "削除",
|
||||||
"nextPage": "次のページ",
|
|
||||||
"copy": "コピー",
|
"copy": "コピー",
|
||||||
"error": "エラー",
|
"error": "エラー",
|
||||||
"file": "ファイル",
|
"file": "ファイル",
|
||||||
@@ -64,13 +62,10 @@
|
|||||||
"input": "インプット",
|
"input": "インプット",
|
||||||
"format": "形式",
|
"format": "形式",
|
||||||
"installed": "インストール済み",
|
"installed": "インストール済み",
|
||||||
"localSystem": "ローカルシステム",
|
|
||||||
"outputs": "アウトプット",
|
"outputs": "アウトプット",
|
||||||
"prevPage": "前のページ",
|
|
||||||
"unknownError": "未知のエラー",
|
"unknownError": "未知のエラー",
|
||||||
"orderBy": "並び順:",
|
"orderBy": "並び順:",
|
||||||
"enabled": "有効",
|
"enabled": "有効",
|
||||||
"notInstalled": "未 $t(common.installed)",
|
|
||||||
"positivePrompt": "ポジティブプロンプト",
|
"positivePrompt": "ポジティブプロンプト",
|
||||||
"negativePrompt": "ネガティブプロンプト",
|
"negativePrompt": "ネガティブプロンプト",
|
||||||
"selected": "選択済み",
|
"selected": "選択済み",
|
||||||
@@ -96,7 +91,6 @@
|
|||||||
"close": "閉じる",
|
"close": "閉じる",
|
||||||
"warnings": "警告",
|
"warnings": "警告",
|
||||||
"dontShowMeThese": "次回から表示しない",
|
"dontShowMeThese": "次回から表示しない",
|
||||||
"goTo": "移動",
|
|
||||||
"generating": "生成中",
|
"generating": "生成中",
|
||||||
"loadingModel": "モデルをロード中",
|
"loadingModel": "モデルをロード中",
|
||||||
"layout": "レイアウト",
|
"layout": "レイアウト",
|
||||||
@@ -107,7 +101,6 @@
|
|||||||
"min": "最小",
|
"min": "最小",
|
||||||
"max": "最大",
|
"max": "最大",
|
||||||
"values": "値",
|
"values": "値",
|
||||||
"resetToDefaults": "デフォルトに戻す",
|
|
||||||
"row": "行",
|
"row": "行",
|
||||||
"column": "列",
|
"column": "列",
|
||||||
"board": "ボード",
|
"board": "ボード",
|
||||||
@@ -131,7 +124,6 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "画像のサイズ",
|
"galleryImageSize": "画像のサイズ",
|
||||||
"gallerySettings": "ギャラリーの設定",
|
"gallerySettings": "ギャラリーの設定",
|
||||||
"noImagesInGallery": "表示する画像がありません",
|
|
||||||
"autoSwitchNewImages": "新しい画像に自動切替",
|
"autoSwitchNewImages": "新しい画像に自動切替",
|
||||||
"copy": "コピー",
|
"copy": "コピー",
|
||||||
"image": "画像",
|
"image": "画像",
|
||||||
@@ -145,7 +137,6 @@
|
|||||||
"deleteImage_other": "画像 {{count}} 枚を削除",
|
"deleteImage_other": "画像 {{count}} 枚を削除",
|
||||||
"deleteImagePermanent": "削除された画像は復元できません。",
|
"deleteImagePermanent": "削除された画像は復元できません。",
|
||||||
"download": "ダウンロード",
|
"download": "ダウンロード",
|
||||||
"unableToLoad": "ギャラリーをロードできません",
|
|
||||||
"bulkDownloadRequested": "ダウンロード準備中",
|
"bulkDownloadRequested": "ダウンロード準備中",
|
||||||
"bulkDownloadRequestedDesc": "ダウンロードの準備中です。しばらくお待ちください。",
|
"bulkDownloadRequestedDesc": "ダウンロードの準備中です。しばらくお待ちください。",
|
||||||
"bulkDownloadRequestFailed": "ダウンロード準備中に問題が発生",
|
"bulkDownloadRequestFailed": "ダウンロード準備中に問題が発生",
|
||||||
@@ -160,7 +151,6 @@
|
|||||||
"compareImage": "比較画像",
|
"compareImage": "比較画像",
|
||||||
"openInViewer": "ビューアで開く",
|
"openInViewer": "ビューアで開く",
|
||||||
"selectForCompare": "比較対象として選択",
|
"selectForCompare": "比較対象として選択",
|
||||||
"selectAnImageToCompare": "比較する画像を選択",
|
|
||||||
"slider": "スライダー",
|
"slider": "スライダー",
|
||||||
"sideBySide": "横並び",
|
"sideBySide": "横並び",
|
||||||
"hover": "ホバー",
|
"hover": "ホバー",
|
||||||
@@ -172,8 +162,6 @@
|
|||||||
"compareHelp4": "<Kbd>[Z</Kbd>]または<Kbd>[Esc</Kbd>]を押して終了します。",
|
"compareHelp4": "<Kbd>[Z</Kbd>]または<Kbd>[Esc</Kbd>]を押して終了します。",
|
||||||
"compareHelp2": "<Kbd>M</Kbd> キーを押して比較モードを切り替えます。",
|
"compareHelp2": "<Kbd>M</Kbd> キーを押して比較モードを切り替えます。",
|
||||||
"move": "移動",
|
"move": "移動",
|
||||||
"openViewer": "ビューアを開く",
|
|
||||||
"closeViewer": "ビューアを閉じる",
|
|
||||||
"exitSearch": "画像検索を終了",
|
"exitSearch": "画像検索を終了",
|
||||||
"oldestFirst": "最古から",
|
"oldestFirst": "最古から",
|
||||||
"showStarredImagesFirst": "スター付き画像を最初に",
|
"showStarredImagesFirst": "スター付き画像を最初に",
|
||||||
@@ -182,7 +170,6 @@
|
|||||||
"searchImages": "メタデータで検索",
|
"searchImages": "メタデータで検索",
|
||||||
"gallery": "ギャラリー",
|
"gallery": "ギャラリー",
|
||||||
"newestFirst": "最新から",
|
"newestFirst": "最新から",
|
||||||
"jump": "ジャンプ",
|
|
||||||
"go": "進む",
|
"go": "進む",
|
||||||
"sortDirection": "並び替え順",
|
"sortDirection": "並び替え順",
|
||||||
"displayBoardSearch": "ボード検索",
|
"displayBoardSearch": "ボード検索",
|
||||||
@@ -325,10 +312,6 @@
|
|||||||
"desc": "リスト内の前のレイヤーを選択します。",
|
"desc": "リスト内の前のレイヤーを選択します。",
|
||||||
"title": "前のレイヤー"
|
"title": "前のレイヤー"
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "ツール色を白に設定",
|
|
||||||
"desc": "現在のツールの色を白色に設定します。"
|
|
||||||
},
|
|
||||||
"selectViewTool": {
|
"selectViewTool": {
|
||||||
"title": "表示ツール",
|
"title": "表示ツール",
|
||||||
"desc": "表示ツールを選択します。"
|
"desc": "表示ツールを選択します。"
|
||||||
@@ -609,7 +592,6 @@
|
|||||||
"scanResults": "結果をスキャン",
|
"scanResults": "結果をスキャン",
|
||||||
"scanPlaceholder": "ローカルフォルダへのパス",
|
"scanPlaceholder": "ローカルフォルダへのパス",
|
||||||
"typePhraseHere": "ここにフレーズを入力",
|
"typePhraseHere": "ここにフレーズを入力",
|
||||||
"ipAdapters": "IPアダプター",
|
|
||||||
"modelImageUpdated": "モデル画像アップデート",
|
"modelImageUpdated": "モデル画像アップデート",
|
||||||
"installAll": "全てインストール",
|
"installAll": "全てインストール",
|
||||||
"installRepo": "リポジトリをインストール",
|
"installRepo": "リポジトリをインストール",
|
||||||
@@ -651,7 +633,6 @@
|
|||||||
"spandrelImageToImage": "Image to Image(スパンドレル)",
|
"spandrelImageToImage": "Image to Image(スパンドレル)",
|
||||||
"starterBundles": "スターターバンドル",
|
"starterBundles": "スターターバンドル",
|
||||||
"starterModels": "スターターモデル",
|
"starterModels": "スターターモデル",
|
||||||
"starterModelsInModelManager": "スターターモデルがモデルマネージャーで見つかりました",
|
|
||||||
"modelImageDeleteFailed": "モデル画像の削除失敗",
|
"modelImageDeleteFailed": "モデル画像の削除失敗",
|
||||||
"urlForbidden": "このモデルにアクセスできません",
|
"urlForbidden": "このモデルにアクセスできません",
|
||||||
"urlForbiddenErrorMessage": "このモデルを配布しているサイトからリクエスト権限が必要かもしれません.",
|
"urlForbiddenErrorMessage": "このモデルを配布しているサイトからリクエスト権限が必要かもしれません.",
|
||||||
@@ -660,12 +641,10 @@
|
|||||||
"inplaceInstall": "定位置にインストール",
|
"inplaceInstall": "定位置にインストール",
|
||||||
"fileSize": "ファイルサイズ",
|
"fileSize": "ファイルサイズ",
|
||||||
"modelPickerFallbackNoModelsInstalled2": "<LinkComponent>モデルマネージャー</LinkComponent> にアクセスしてモデルをインストールしてください.",
|
"modelPickerFallbackNoModelsInstalled2": "<LinkComponent>モデルマネージャー</LinkComponent> にアクセスしてモデルをインストールしてください.",
|
||||||
"filterModels": "フィルターモデル",
|
|
||||||
"modelPickerFallbackNoModelsInstalled": "モデルがインストールされていません.",
|
"modelPickerFallbackNoModelsInstalled": "モデルがインストールされていません.",
|
||||||
"manageModels": "モデル管理",
|
"manageModels": "モデル管理",
|
||||||
"hfTokenReset": "ハギングフェイストークンリセット",
|
"hfTokenReset": "ハギングフェイストークンリセット",
|
||||||
"relatedModels": "関連のあるモデル",
|
"relatedModels": "関連のあるモデル",
|
||||||
"showOnlyRelatedModels": "関連している",
|
|
||||||
"installedModelsCount": "{{total}} モデルのうち {{installed}} 個がインストールされています。",
|
"installedModelsCount": "{{total}} モデルのうち {{installed}} 個がインストールされています。",
|
||||||
"allNModelsInstalled": "{{count}} 個のモデルがすべてインストールされています",
|
"allNModelsInstalled": "{{count}} 個のモデルがすべてインストールされています",
|
||||||
"nToInstall": "{{count}}個をインストールする",
|
"nToInstall": "{{count}}個をインストールする",
|
||||||
@@ -682,12 +661,8 @@
|
|||||||
"scanFolderDescription": "ローカルフォルダをスキャンしてモデルを自動的に検出し、インストールします。",
|
"scanFolderDescription": "ローカルフォルダをスキャンしてモデルを自動的に検出し、インストールします。",
|
||||||
"recommendedModels": "推奨モデル",
|
"recommendedModels": "推奨モデル",
|
||||||
"exploreStarter": "または、利用可能なすべてのスターターモデルを参照してください",
|
"exploreStarter": "または、利用可能なすべてのスターターモデルを参照してください",
|
||||||
"quickStart": "クイックスタートバンドル",
|
|
||||||
"bundleDescription": "各バンドルには各モデルファミリーの必須モデルと、開始するための厳選されたベースモデルが含まれています。",
|
"bundleDescription": "各バンドルには各モデルファミリーの必須モデルと、開始するための厳選されたベースモデルが含まれています。",
|
||||||
"browseAll": "または、利用可能なすべてのモデルを参照してください。",
|
"sdxl": "SDXL"
|
||||||
"stableDiffusion15": "Stable Diffusion1.5",
|
|
||||||
"sdxl": "SDXL",
|
|
||||||
"fluxDev": "FLUX.1 dev"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"parameters": {
|
"parameters": {
|
||||||
@@ -703,12 +678,10 @@
|
|||||||
"scaleBeforeProcessing": "処理前のスケール",
|
"scaleBeforeProcessing": "処理前のスケール",
|
||||||
"scaledWidth": "幅のスケール",
|
"scaledWidth": "幅のスケール",
|
||||||
"scaledHeight": "高さのスケール",
|
"scaledHeight": "高さのスケール",
|
||||||
"downloadImage": "画像をダウンロード",
|
|
||||||
"usePrompt": "プロンプトを使用",
|
"usePrompt": "プロンプトを使用",
|
||||||
"useSeed": "シード値を使用",
|
"useSeed": "シード値を使用",
|
||||||
"useAll": "すべてを使用",
|
"useAll": "すべてを使用",
|
||||||
"info": "情報",
|
"info": "情報",
|
||||||
"showOptionsPanel": "サイドパネルを表示 (O or T)",
|
|
||||||
"iterations": "生成回数",
|
"iterations": "生成回数",
|
||||||
"general": "基本設定",
|
"general": "基本設定",
|
||||||
"setToOptimalSize": "サイズをモデルに最適化",
|
"setToOptimalSize": "サイズをモデルに最適化",
|
||||||
@@ -722,7 +695,6 @@
|
|||||||
"collectionNumberLTExclusiveMin": "{{value}} <= {{exclusiveMinimum}} (最小値を除く)",
|
"collectionNumberLTExclusiveMin": "{{value}} <= {{exclusiveMinimum}} (最小値を除く)",
|
||||||
"missingInputForField": "入力の欠落",
|
"missingInputForField": "入力の欠落",
|
||||||
"noModelSelected": "モデルが選択されていません",
|
"noModelSelected": "モデルが選択されていません",
|
||||||
"emptyBatches": "空のバッチ",
|
|
||||||
"collectionStringTooLong": "長すぎます,最大{{maxLength}}",
|
"collectionStringTooLong": "長すぎます,最大{{maxLength}}",
|
||||||
"batchNodeCollectionSizeMismatchNoGroupId": "バッチグループのコレクションサイズが合いません",
|
"batchNodeCollectionSizeMismatchNoGroupId": "バッチグループのコレクションサイズが合いません",
|
||||||
"invoke": "呼び出す",
|
"invoke": "呼び出す",
|
||||||
@@ -734,7 +706,6 @@
|
|||||||
"missingNodeTemplate": "ノードテンプレートの欠落",
|
"missingNodeTemplate": "ノードテンプレートの欠落",
|
||||||
"batchNodeNotConnected": "バッチノードが: {{label}}につながっていない",
|
"batchNodeNotConnected": "バッチノードが: {{label}}につながっていない",
|
||||||
"collectionNumberLTMin": "{{value}} < {{minimum}} (最小増加)",
|
"collectionNumberLTMin": "{{value}} < {{minimum}} (最小増加)",
|
||||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), スケーリングされたbboxの高さは{{height}}です",
|
|
||||||
"fluxModelMultipleControlLoRAs": "コントロールLoRAは1度に1つしか使用できません",
|
"fluxModelMultipleControlLoRAs": "コントロールLoRAは1度に1つしか使用できません",
|
||||||
"noPrompts": "プロンプトが生成されません",
|
"noPrompts": "プロンプトが生成されません",
|
||||||
"noNodesInGraph": "グラフにノードがありません",
|
"noNodesInGraph": "グラフにノードがありません",
|
||||||
@@ -742,7 +713,6 @@
|
|||||||
"canvasIsFiltering": "キャンバスがビジー状態(フィルタリング)",
|
"canvasIsFiltering": "キャンバスがビジー状態(フィルタリング)",
|
||||||
"canvasIsCompositing": "キャンバスがビジー状態(合成)",
|
"canvasIsCompositing": "キャンバスがビジー状態(合成)",
|
||||||
"systemDisconnected": "システムが切断されました",
|
"systemDisconnected": "システムが切断されました",
|
||||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), 拡大縮小されたbboxの幅は{{width}}です",
|
|
||||||
"canvasIsTransforming": "キャンバスがビジー状態(変換)",
|
"canvasIsTransforming": "キャンバスがビジー状態(変換)",
|
||||||
"canvasIsRasterizing": "キャンバスがビジー状態(ラスタライズ)",
|
"canvasIsRasterizing": "キャンバスがビジー状態(ラスタライズ)",
|
||||||
"modelIncompatibleBboxHeight": "Bboxの高さは{{height}}ですが,{{model}}は{{multiple}}の倍数が必要です",
|
"modelIncompatibleBboxHeight": "Bboxの高さは{{height}}ですが,{{model}}は{{multiple}}の倍数が必要です",
|
||||||
@@ -750,8 +720,6 @@
|
|||||||
"modelIncompatibleBboxWidth": "Bboxの幅は{{width}}ですが, {{model}}は{{multiple}}の倍数が必要です",
|
"modelIncompatibleBboxWidth": "Bboxの幅は{{width}}ですが, {{model}}は{{multiple}}の倍数が必要です",
|
||||||
"modelIncompatibleScaledBboxWidth": "bboxの幅は{{width}}ですが,{{model}}は{{multiple}}の倍数が必要です",
|
"modelIncompatibleScaledBboxWidth": "bboxの幅は{{width}}ですが,{{model}}は{{multiple}}の倍数が必要です",
|
||||||
"canvasIsSelectingObject": "キャンバスがビジー状態(オブジェクトの選択)",
|
"canvasIsSelectingObject": "キャンバスがビジー状態(オブジェクトの選択)",
|
||||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), bboxの幅は{{width}}です",
|
|
||||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), bboxの高さは{{height}}です",
|
|
||||||
"noFLUXVAEModelSelected": "FLUX生成にVAEモデルが選択されていません",
|
"noFLUXVAEModelSelected": "FLUX生成にVAEモデルが選択されていません",
|
||||||
"noT5EncoderModelSelected": "FLUX生成にT5エンコーダモデルが選択されていません",
|
"noT5EncoderModelSelected": "FLUX生成にT5エンコーダモデルが選択されていません",
|
||||||
"modelDisabledForTrial": "{{modelName}} を使用した生成はトライアルアカウントではご利用いただけません.アカウント設定にアクセスしてアップグレードしてください。",
|
"modelDisabledForTrial": "{{modelName}} を使用した生成はトライアルアカウントではご利用いただけません.アカウント設定にアクセスしてアップグレードしてください。",
|
||||||
@@ -829,8 +797,6 @@
|
|||||||
"enableHighlightFocusedRegions": "重点領域を強調表示",
|
"enableHighlightFocusedRegions": "重点領域を強調表示",
|
||||||
"clearIntermediatesDesc1": "中間物をクリアすると、キャンバスとコントロールネットの状態がリセットされます.",
|
"clearIntermediatesDesc1": "中間物をクリアすると、キャンバスとコントロールネットの状態がリセットされます.",
|
||||||
"showProgressInViewer": "ビューアで進行状況画像を表示する",
|
"showProgressInViewer": "ビューアで進行状況画像を表示する",
|
||||||
"modelDescriptionsDisabled": "ドロップダウンのモデル説明が無効になっています",
|
|
||||||
"modelDescriptionsDisabledDesc": "ドロップダウンのモデル説明が無効になっています.設定で有効にしてください.",
|
|
||||||
"clearIntermediatesDisabled": "中間物をクリアするにはキューが空でなければなりません",
|
"clearIntermediatesDisabled": "中間物をクリアするにはキューが空でなければなりません",
|
||||||
"clearIntermediatesDesc2": "中間画像は生成時に生成される副産物であり、ギャラリーに表示される結果画像とは異なります.中間画像を削除するとディスク容量が解放されます.",
|
"clearIntermediatesDesc2": "中間画像は生成時に生成される副産物であり、ギャラリーに表示される結果画像とは異なります.中間画像を削除するとディスク容量が解放されます.",
|
||||||
"intermediatesClearedFailed": "中間物をクリアする問題",
|
"intermediatesClearedFailed": "中間物をクリアする問題",
|
||||||
@@ -861,11 +827,9 @@
|
|||||||
"imagesWillBeAddedTo": "アップロードされた画像はボード {{boardName}} のアセットに追加されます.",
|
"imagesWillBeAddedTo": "アップロードされた画像はボード {{boardName}} のアセットに追加されます.",
|
||||||
"layerCopiedToClipboard": "レイヤーがクリップボードにコピーされました",
|
"layerCopiedToClipboard": "レイヤーがクリップボードにコピーされました",
|
||||||
"pasteFailed": "貼り付け失敗",
|
"pasteFailed": "貼り付け失敗",
|
||||||
"imageSavingFailed": "画像保存に失敗しました",
|
|
||||||
"importSuccessful": "インポートが成功しました",
|
"importSuccessful": "インポートが成功しました",
|
||||||
"problemDownloadingImage": "画像をダウンロードできません",
|
"problemDownloadingImage": "画像をダウンロードできません",
|
||||||
"modelAddedSimple": "モデルがキューに追加されました",
|
"modelAddedSimple": "モデルがキューに追加されました",
|
||||||
"uploadFailedInvalidUploadDesc_withCount_other": "PNG、JPEG、または WEBP 画像は最大 1 つにする必要があります.",
|
|
||||||
"outOfMemoryErrorDesc": "現在の生成設定はシステム容量を超えています.設定を調整してもう一度お試しください.",
|
"outOfMemoryErrorDesc": "現在の生成設定はシステム容量を超えています.設定を調整してもう一度お試しください.",
|
||||||
"parametersSet": "パラメーターが呼び出されました",
|
"parametersSet": "パラメーターが呼び出されました",
|
||||||
"modelImportCanceled": "モデルのインポートがキャンセルされました",
|
"modelImportCanceled": "モデルのインポートがキャンセルされました",
|
||||||
@@ -880,14 +844,11 @@
|
|||||||
"linkCopied": "リンクがコピーされました",
|
"linkCopied": "リンクがコピーされました",
|
||||||
"unableToLoadImage": "画像をロードできません",
|
"unableToLoadImage": "画像をロードできません",
|
||||||
"unableToLoadImageMetadata": "画像のメタデータをロードできません",
|
"unableToLoadImageMetadata": "画像のメタデータをロードできません",
|
||||||
"imageSaved": "画像が保存されました",
|
|
||||||
"importFailed": "インポートに失敗しました",
|
"importFailed": "インポートに失敗しました",
|
||||||
"invalidUpload": "無効なアップロードです",
|
|
||||||
"outOfMemoryError": "メモリ不足エラー",
|
"outOfMemoryError": "メモリ不足エラー",
|
||||||
"parameterSetDesc": "{{parameter}}を呼び出し",
|
"parameterSetDesc": "{{parameter}}を呼び出し",
|
||||||
"errorCopied": "エラーがコピーされました",
|
"errorCopied": "エラーがコピーされました",
|
||||||
"sentToCanvas": "キャンバスに送信",
|
"sentToCanvas": "キャンバスに送信",
|
||||||
"setControlImage": "コントロール画像としてセット",
|
|
||||||
"workflowLoaded": "ワークフローがロードされました",
|
"workflowLoaded": "ワークフローがロードされました",
|
||||||
"unableToCopy": "コピーできません",
|
"unableToCopy": "コピーできません",
|
||||||
"unableToCopyDesc": "あなたのブラウザはクリップボードアクセスをサポートしていません.Firefoxユーザーの場合は、以下の手順で修正できる可能性があります. ",
|
"unableToCopyDesc": "あなたのブラウザはクリップボードアクセスをサポートしていません.Firefoxユーザーの場合は、以下の手順で修正できる可能性があります. ",
|
||||||
@@ -901,32 +862,23 @@
|
|||||||
"parameterNotSetDescWithMessage": "{{parameter}}: {{message}}を呼び出せません",
|
"parameterNotSetDescWithMessage": "{{parameter}}: {{message}}を呼び出せません",
|
||||||
"problemCopyingLayer": "レイヤーをコピーできません",
|
"problemCopyingLayer": "レイヤーをコピーできません",
|
||||||
"problemSavingLayer": "レイヤー保存ができません",
|
"problemSavingLayer": "レイヤー保存ができません",
|
||||||
"setNodeField": "ノードフィールドとしてセット",
|
|
||||||
"layerSavedToAssets": "レイヤーがアセットに保存されました",
|
|
||||||
"outOfMemoryErrorDescLocal": "OOM を削減するには、<LinkComponent>低 VRAM ガイド</LinkComponent> に従ってください.",
|
"outOfMemoryErrorDescLocal": "OOM を削減するには、<LinkComponent>低 VRAM ガイド</LinkComponent> に従ってください.",
|
||||||
"parameterNotSet": "パラメーターが呼び出されていません",
|
"parameterNotSet": "パラメーターが呼び出されていません",
|
||||||
"addedToBoard": "{{name}} 個の資産をボードに追加しました",
|
"addedToBoard": "{{name}} 個の資産をボードに追加しました",
|
||||||
"addedToUncategorized": "$t(boards.uncategorized)個のアセットがボードに追加されました",
|
"addedToUncategorized": "$t(boards.uncategorized)個のアセットがボードに追加されました",
|
||||||
"problemDeletingWorkflow": "ワークフローが削除された問題",
|
"problemDeletingWorkflow": "ワークフローが削除された問題",
|
||||||
"imageNotLoadedDesc": "画像を見つけられません",
|
|
||||||
"parameterNotSetDesc": "{{parameter}}を呼び出せません",
|
"parameterNotSetDesc": "{{parameter}}を呼び出せません",
|
||||||
"chatGPT4oIncompatibleGenerationMode": "ChatGPT 4oは,テキストから画像への生成と画像から画像への生成のみをサポートしています.インペインティングおよび,アウトペインティングタスクには他のモデルを使用してください.",
|
"chatGPT4oIncompatibleGenerationMode": "ChatGPT 4oは,テキストから画像への生成と画像から画像への生成のみをサポートしています.インペインティングおよび,アウトペインティングタスクには他のモデルを使用してください.",
|
||||||
"imagenIncompatibleGenerationMode": "Google {{model}} はテキストから画像への変換のみをサポートしています. 画像から画像への変換, インペインティング,アウトペインティングのタスクには他のモデルを使用してください.",
|
"imagenIncompatibleGenerationMode": "Google {{model}} はテキストから画像への変換のみをサポートしています. 画像から画像への変換, インペインティング,アウトペインティングのタスクには他のモデルを使用してください.",
|
||||||
"noRasterLayers": "ラスターレイヤーが見つかりません",
|
|
||||||
"noRasterLayersDesc": "PSDにエクスポートするには、少なくとも1つのラスターレイヤーを作成します",
|
|
||||||
"noActiveRasterLayers": "アクティブなラスターレイヤーがありません",
|
|
||||||
"noActiveRasterLayersDesc": "PSD にエクスポートするには、少なくとも 1 つのラスター レイヤーを有効にします",
|
|
||||||
"noVisibleRasterLayers": "表示されるラスター レイヤーがありません",
|
"noVisibleRasterLayers": "表示されるラスター レイヤーがありません",
|
||||||
"noVisibleRasterLayersDesc": "PSD にエクスポートするには、少なくとも 1 つのラスター レイヤーを有効にします",
|
"noVisibleRasterLayersDesc": "PSD にエクスポートするには、少なくとも 1 つのラスター レイヤーを有効にします",
|
||||||
"invalidCanvasDimensions": "キャンバスのサイズが無効です",
|
"invalidCanvasDimensions": "キャンバスのサイズが無効です",
|
||||||
"canvasTooLarge": "キャンバスが大きすぎます",
|
"canvasTooLarge": "キャンバスが大きすぎます",
|
||||||
"canvasTooLargeDesc": "キャンバスのサイズがPSDエクスポートの最大許容サイズを超えています。キャンバス全体の幅と高さを小さくしてから、もう一度お試しください。",
|
"canvasTooLargeDesc": "キャンバスのサイズがPSDエクスポートの最大許容サイズを超えています。キャンバス全体の幅と高さを小さくしてから、もう一度お試しください。",
|
||||||
"failedToProcessLayers": "レイヤーの処理に失敗しました",
|
|
||||||
"psdExportSuccess": "PSDエクスポート完了",
|
"psdExportSuccess": "PSDエクスポート完了",
|
||||||
"psdExportSuccessDesc": "{{count}} 個のレイヤーを PSD ファイルに正常にエクスポートしました",
|
"psdExportSuccessDesc": "{{count}} 個のレイヤーを PSD ファイルに正常にエクスポートしました",
|
||||||
"problemExportingPSD": "PSD のエクスポート中に問題が発生しました",
|
"problemExportingPSD": "PSD のエクスポート中に問題が発生しました",
|
||||||
"canvasManagerNotAvailable": "キャンバスマネージャーは利用できません",
|
"canvasManagerNotAvailable": "キャンバスマネージャーは利用できません",
|
||||||
"noValidLayerAdapters": "有効なレイヤーアダプタが見つかりません",
|
|
||||||
"fluxKontextIncompatibleGenerationMode": "Flux Kontext はテキストから画像への変換のみをサポートしています。画像から画像への変換、インペインティング、アウトペインティングのタスクには他のモデルを使用してください。",
|
"fluxKontextIncompatibleGenerationMode": "Flux Kontext はテキストから画像への変換のみをサポートしています。画像から画像への変換、インペインティング、アウトペインティングのタスクには他のモデルを使用してください。",
|
||||||
"promptGenerationStarted": "プロンプト生成が開始されました",
|
"promptGenerationStarted": "プロンプト生成が開始されました",
|
||||||
"uploadAndPromptGenerationFailed": "画像のアップロードとプロンプトの生成に失敗しました",
|
"uploadAndPromptGenerationFailed": "画像のアップロードとプロンプトの生成に失敗しました",
|
||||||
@@ -958,7 +910,6 @@
|
|||||||
"positivePrompt": "ポジティブプロンプト",
|
"positivePrompt": "ポジティブプロンプト",
|
||||||
"strength": "Image to Image 強度",
|
"strength": "Image to Image 強度",
|
||||||
"recallParameters": "パラメータを再使用",
|
"recallParameters": "パラメータを再使用",
|
||||||
"recallParameter": "{{label}} を再使用",
|
|
||||||
"imageDimensions": "画像サイズ",
|
"imageDimensions": "画像サイズ",
|
||||||
"imageDetails": "画像の詳細",
|
"imageDetails": "画像の詳細",
|
||||||
"model": "モデル",
|
"model": "モデル",
|
||||||
@@ -973,7 +924,6 @@
|
|||||||
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
||||||
"canvasV2Metadata": "キャンバス",
|
"canvasV2Metadata": "キャンバス",
|
||||||
"guidance": "手引き",
|
"guidance": "手引き",
|
||||||
"parsingFailed": "解析に失敗しました",
|
|
||||||
"seamlessXAxis": "シームレスX軸",
|
"seamlessXAxis": "シームレスX軸",
|
||||||
"seamlessYAxis": "シームレスY軸",
|
"seamlessYAxis": "シームレスY軸",
|
||||||
"parameterSet": "パラメーター {{parameter}} が設定されました",
|
"parameterSet": "パラメーター {{parameter}} が設定されました",
|
||||||
@@ -1031,7 +981,6 @@
|
|||||||
"clearQueueAlertDialog2": "キューをクリアしてもよろしいですか?",
|
"clearQueueAlertDialog2": "キューをクリアしてもよろしいですか?",
|
||||||
"item": "項目",
|
"item": "項目",
|
||||||
"graphFailedToQueue": "グラフをキューに追加できませんでした",
|
"graphFailedToQueue": "グラフをキューに追加できませんでした",
|
||||||
"batchFieldValues": "バッチの詳細",
|
|
||||||
"openQueue": "キューを開く",
|
"openQueue": "キューを開く",
|
||||||
"time": "時間",
|
"time": "時間",
|
||||||
"completedIn": "完了まで",
|
"completedIn": "完了まで",
|
||||||
@@ -1061,14 +1010,12 @@
|
|||||||
"models": {
|
"models": {
|
||||||
"noMatchingModels": "一致するモデルがありません",
|
"noMatchingModels": "一致するモデルがありません",
|
||||||
"loading": "読み込み中",
|
"loading": "読み込み中",
|
||||||
"noMatchingLoRAs": "一致するLoRAがありません",
|
|
||||||
"noModelsAvailable": "使用可能なモデルがありません",
|
"noModelsAvailable": "使用可能なモデルがありません",
|
||||||
"selectModel": "モデルを選択してください",
|
"selectModel": "モデルを選択してください",
|
||||||
"concepts": "コンセプト",
|
"concepts": "コンセプト",
|
||||||
"addLora": "LoRAを追加",
|
"addLora": "LoRAを追加",
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
"defaultVAE": "デフォルトVAE",
|
"defaultVAE": "デフォルトVAE",
|
||||||
"noLoRAsInstalled": "インストールされているLoRAはありません",
|
|
||||||
"noRefinerModelsInstalled": "インストールされているSDXLリファイナーモデルはありません",
|
"noRefinerModelsInstalled": "インストールされているSDXLリファイナーモデルはありません",
|
||||||
"noCompatibleLoRAs": "互換性のあるLoRAはありません"
|
"noCompatibleLoRAs": "互換性のあるLoRAはありません"
|
||||||
},
|
},
|
||||||
@@ -1078,7 +1025,6 @@
|
|||||||
"addNodeToolTip": "ノードを追加 (Shift+A, Space)",
|
"addNodeToolTip": "ノードを追加 (Shift+A, Space)",
|
||||||
"missingTemplate": "Invalid node: タイプ {{type}} のノード {{node}} にテンプレートがありません(未インストール?)",
|
"missingTemplate": "Invalid node: タイプ {{type}} のノード {{node}} にテンプレートがありません(未インストール?)",
|
||||||
"loadWorkflow": "ワークフローを読み込み",
|
"loadWorkflow": "ワークフローを読み込み",
|
||||||
"hideLegendNodes": "フィールドタイプの凡例を非表示",
|
|
||||||
"float": "浮動小数点",
|
"float": "浮動小数点",
|
||||||
"integer": "整数",
|
"integer": "整数",
|
||||||
"nodeTemplate": "ノードテンプレート",
|
"nodeTemplate": "ノードテンプレート",
|
||||||
@@ -1122,13 +1068,11 @@
|
|||||||
"enum": "Enum",
|
"enum": "Enum",
|
||||||
"arithmeticSequence": "等差数列",
|
"arithmeticSequence": "等差数列",
|
||||||
"linearDistribution": "線形分布",
|
"linearDistribution": "線形分布",
|
||||||
"addLinearView": "ライナービューに追加",
|
|
||||||
"animatedEdges": "アニメーションエッジ",
|
"animatedEdges": "アニメーションエッジ",
|
||||||
"uniformRandomDistribution": "一様ランダム分布",
|
"uniformRandomDistribution": "一様ランダム分布",
|
||||||
"noBatchGroup": "グループなし",
|
"noBatchGroup": "グループなし",
|
||||||
"parseString": "文字列の解析",
|
"parseString": "文字列の解析",
|
||||||
"generatorImagesFromBoard": "ボードからの画像",
|
"generatorImagesFromBoard": "ボードからの画像",
|
||||||
"generatorLoading": "読み込み中",
|
|
||||||
"missingNode": "呼び出しノードがありません",
|
"missingNode": "呼び出しノードがありません",
|
||||||
"missingSourceOrTargetNode": "ソースまたはターゲットノードがありません",
|
"missingSourceOrTargetNode": "ソースまたはターゲットノードがありません",
|
||||||
"missingSourceOrTargetHandle": "ソースまたはターゲットハンドルがありません",
|
"missingSourceOrTargetHandle": "ソースまたはターゲットハンドルがありません",
|
||||||
@@ -1149,7 +1093,6 @@
|
|||||||
"missingInvocationTemplate": "呼び出しテンプレートがありません",
|
"missingInvocationTemplate": "呼び出しテンプレートがありません",
|
||||||
"nodePack": "ノードパック",
|
"nodePack": "ノードパック",
|
||||||
"targetNodeFieldDoesNotExist": "無効なエッジ:ターゲット/インプットフィールド{{node}}.{{field}} が存在しません",
|
"targetNodeFieldDoesNotExist": "無効なエッジ:ターゲット/インプットフィールド{{node}}.{{field}} が存在しません",
|
||||||
"mismatchedVersion": "無効なノード:ノード {{node}} のタイプ {{type}} はバージョンとミスマッチしています (アップデートを試されますか?)",
|
|
||||||
"dynamicPromptsCombinatorial": "ダイナミックプロンプト(組み合わせ)",
|
"dynamicPromptsCombinatorial": "ダイナミックプロンプト(組み合わせ)",
|
||||||
"cannotMixAndMatchCollectionItemTypes": "コレクション・アイテムの種類を組み合わせることはできません",
|
"cannotMixAndMatchCollectionItemTypes": "コレクション・アイテムの種類を組み合わせることはできません",
|
||||||
"missingFieldTemplate": "フィールドテンプレートがありません",
|
"missingFieldTemplate": "フィールドテンプレートがありません",
|
||||||
@@ -1159,7 +1102,6 @@
|
|||||||
"collectionOrScalarFieldType": "{{name}} (単数またはコレクション)",
|
"collectionOrScalarFieldType": "{{name}} (単数またはコレクション)",
|
||||||
"unableToUpdateNode": "ノードアップロード失敗:ノード {{node}} のタイプ {{type}} (削除か再生成が必要かもしれません)",
|
"unableToUpdateNode": "ノードアップロード失敗:ノード {{node}} のタイプ {{type}} (削除か再生成が必要かもしれません)",
|
||||||
"deletedInvalidEdge": "無効なエッジを削除しました{{source}} -> {{target}}",
|
"deletedInvalidEdge": "無効なエッジを削除しました{{source}} -> {{target}}",
|
||||||
"noFieldsLinearview": "線形ビューに追加されたフィールドがありません",
|
|
||||||
"collectionFieldType": "{{name}} (コレクション)",
|
"collectionFieldType": "{{name}} (コレクション)",
|
||||||
"colorCodeEdgesHelp": "接続されたフィールドによるカラーコードエッジ",
|
"colorCodeEdgesHelp": "接続されたフィールドによるカラーコードエッジ",
|
||||||
"showEdgeLabelsHelp": "エッジのラベルを表示,接続されているノードを示す",
|
"showEdgeLabelsHelp": "エッジのラベルを表示,接続されているノードを示す",
|
||||||
@@ -1174,7 +1116,6 @@
|
|||||||
"loadWorkflowDesc2": "現在のワークフローは保存されていない変更があります.",
|
"loadWorkflowDesc2": "現在のワークフローは保存されていない変更があります.",
|
||||||
"clearWorkflowDesc": "このワークフローをクリアして新しいワークフローにしますか?",
|
"clearWorkflowDesc": "このワークフローをクリアして新しいワークフローにしますか?",
|
||||||
"updateNode": "ノードをアップデート",
|
"updateNode": "ノードをアップデート",
|
||||||
"versionUnknown": " バージョン不明",
|
|
||||||
"graph": "グラフ",
|
"graph": "グラフ",
|
||||||
"workflowContact": "お問い合わせ",
|
"workflowContact": "お問い合わせ",
|
||||||
"outputFieldTypeParseError": "出力フィールド {{node}}.{{field}} の型を解析できません({{message}})",
|
"outputFieldTypeParseError": "出力フィールド {{node}}.{{field}} の型を解析できません({{message}})",
|
||||||
@@ -1193,36 +1134,28 @@
|
|||||||
"unableToExtractSchemaNameFromRef": "参照からスキーマ名を抽出できません",
|
"unableToExtractSchemaNameFromRef": "参照からスキーマ名を抽出できません",
|
||||||
"unableToUpdateNodes_other": "{{count}} 個のノードをアップデートできません",
|
"unableToUpdateNodes_other": "{{count}} 個のノードをアップデートできません",
|
||||||
"workflowSettings": "ワークフローエディター設定",
|
"workflowSettings": "ワークフローエディター設定",
|
||||||
"generateValues": "値を生成",
|
|
||||||
"floatRangeGenerator": "浮動小数点レンジ生成器",
|
|
||||||
"integerRangeGenerator": "整数レンジ生成器",
|
|
||||||
"specialDesc": "この呼び出しは,アプリ内で特別な処理を行います.例えば,バッチノードは1つのワークフローから複数のグラフをキューに入れるために使用されます.",
|
"specialDesc": "この呼び出しは,アプリ内で特別な処理を行います.例えば,バッチノードは1つのワークフローから複数のグラフをキューに入れるために使用されます.",
|
||||||
"modelAccessError": "モデル {{key}}が見つからないので,デフォルトにリセットします",
|
"modelAccessError": "モデル {{key}}が見つからないので,デフォルトにリセットします",
|
||||||
"betaDesc": "この呼び出しはベータ版です.安定するまでは,アプリのアップデートの際に変更される可能性があります.この呼び出しは長期的にサポートする予定です.",
|
"betaDesc": "この呼び出しはベータ版です.安定するまでは,アプリのアップデートの際に変更される可能性があります.この呼び出しは長期的にサポートする予定です.",
|
||||||
"internalDesc": "この呼び出しはInvokeによって内部的に使用されます.アプリの更新時に変更される可能性があり,いつでも削除される可能性があります.",
|
"internalDesc": "この呼び出しはInvokeによって内部的に使用されます.アプリの更新時に変更される可能性があり,いつでも削除される可能性があります.",
|
||||||
"noFieldsViewMode": "このワークフローには表示する選択フィールドがありません.値を設定するためにはワークフロー全体を表示します.",
|
"noFieldsViewMode": "このワークフローには表示する選択フィールドがありません.値を設定するためにはワークフロー全体を表示します.",
|
||||||
"clearWorkflow": "ワークフローをクリア",
|
"clearWorkflow": "ワークフローをクリア",
|
||||||
"removeLinearView": "線形ビューから削除",
|
|
||||||
"snapToGrid": "グリッドにスナップ",
|
"snapToGrid": "グリッドにスナップ",
|
||||||
"showMinimapnodes": "ミニマップを表示",
|
"showMinimapnodes": "ミニマップを表示",
|
||||||
"reorderLinearView": "線形ビューの並び替え",
|
|
||||||
"description": "説明",
|
"description": "説明",
|
||||||
"notesDescription": "ワークフローに関するメモを追加する",
|
"notesDescription": "ワークフローに関するメモを追加する",
|
||||||
"newWorkflowDesc2": "現在のワークフローに保存されていない変更があります.",
|
"newWorkflowDesc2": "現在のワークフローに保存されていない変更があります.",
|
||||||
"unknownField": "不明なフィールド",
|
"unknownField": "不明なフィールド",
|
||||||
"unexpectedField_withName": "予期しないフィールド\"{{name}}\"",
|
"unexpectedField_withName": "予期しないフィールド\"{{name}}\"",
|
||||||
"loadingTemplates": "読み込み中 {{name}}",
|
|
||||||
"validateConnectionsHelp": "無効な接続が行われたり,無効なグラフが呼び出されたりしないようにします",
|
"validateConnectionsHelp": "無効な接続が行われたり,無効なグラフが呼び出されたりしないようにします",
|
||||||
"validateConnections": "接続とグラフを確認する",
|
"validateConnections": "接続とグラフを確認する",
|
||||||
"saveToGallery": "ギャラリーに保存",
|
"saveToGallery": "ギャラリーに保存",
|
||||||
"newWorkflowDesc": "新しいワークフローを作りますか?",
|
"newWorkflowDesc": "新しいワークフローを作りますか?",
|
||||||
"unknownFieldType": "$t(nodes.unknownField)型: {{type}}",
|
"unknownFieldType": "$t(nodes.unknownField)型: {{type}}",
|
||||||
"unsupportedArrayItemType": "サポートされていない配列項目型です \"{{type}}\"",
|
"unsupportedArrayItemType": "サポートされていない配列項目型です \"{{type}}\"",
|
||||||
"unableToLoadWorkflow": "ワークフローが読み込めません",
|
|
||||||
"unableToValidateWorkflow": "ワークフローを確認できません",
|
"unableToValidateWorkflow": "ワークフローを確認できません",
|
||||||
"unknownErrorValidatingWorkflow": "ワークフローの確認で不明なエラーが発生",
|
"unknownErrorValidatingWorkflow": "ワークフローの確認で不明なエラーが発生",
|
||||||
"clearWorkflowDesc2": "現在のワークフローは保存されていない変更があります.",
|
"clearWorkflowDesc2": "現在のワークフローは保存されていない変更があります.",
|
||||||
"showLegendNodes": "フィールドタイプの凡例を表示",
|
|
||||||
"unsupportedMismatchedUnion": "CollectionOrScalar型とベース型{{firstType}}および{{secondType}}が不一致です",
|
"unsupportedMismatchedUnion": "CollectionOrScalar型とベース型{{firstType}}および{{secondType}}が不一致です",
|
||||||
"updateApp": "アプリケーションをアップデート",
|
"updateApp": "アプリケーションをアップデート",
|
||||||
"noGraph": "グラフなし",
|
"noGraph": "グラフなし",
|
||||||
@@ -1240,10 +1173,8 @@
|
|||||||
"workflowDescription": "短い説明",
|
"workflowDescription": "短い説明",
|
||||||
"workflowValidation": "ワークフロー検証エラー",
|
"workflowValidation": "ワークフロー検証エラー",
|
||||||
"noOutputRecorded": "記録されたアウトプットがありません",
|
"noOutputRecorded": "記録されたアウトプットがありません",
|
||||||
"unknownTemplate": "不明なテンプレート",
|
|
||||||
"nodeOpacity": "ノードの不透明度",
|
"nodeOpacity": "ノードの不透明度",
|
||||||
"unableToParseFieldType": "フィールドタイプを解析できません",
|
"unableToParseFieldType": "フィールドタイプを解析できません"
|
||||||
"unknownInput": "不明な入力: {{name}}"
|
|
||||||
},
|
},
|
||||||
"boards": {
|
"boards": {
|
||||||
"autoAddBoard": "自動追加するボード",
|
"autoAddBoard": "自動追加するボード",
|
||||||
@@ -1267,7 +1198,6 @@
|
|||||||
"deleteBoardOnly": "ボードのみ削除",
|
"deleteBoardOnly": "ボードのみ削除",
|
||||||
"deletedBoardsCannotbeRestored": "削除したボードと画像は復元できません。「ボードのみ削除」を選択すると、画像は未分類の状態になります。",
|
"deletedBoardsCannotbeRestored": "削除したボードと画像は復元できません。「ボードのみ削除」を選択すると、画像は未分類の状態になります。",
|
||||||
"movingImagesToBoard_other": "{{count}} の画像をボードに移動:",
|
"movingImagesToBoard_other": "{{count}} の画像をボードに移動:",
|
||||||
"hideBoards": "ボードを隠す",
|
|
||||||
"assetsWithCount_other": "{{count}} のアセット",
|
"assetsWithCount_other": "{{count}} のアセット",
|
||||||
"addPrivateBoard": "プライベートボードを追加",
|
"addPrivateBoard": "プライベートボードを追加",
|
||||||
"addSharedBoard": "共有ボードを追加",
|
"addSharedBoard": "共有ボードを追加",
|
||||||
@@ -1282,10 +1212,8 @@
|
|||||||
"selectedForAutoAdd": "自動追加に選択済み",
|
"selectedForAutoAdd": "自動追加に選択済み",
|
||||||
"deletedPrivateBoardsCannotbeRestored": "削除されたボードと画像は復元できません。「ボードのみ削除」を選択すると、画像は作成者に対して非公開の未分類状態になります。",
|
"deletedPrivateBoardsCannotbeRestored": "削除されたボードと画像は復元できません。「ボードのみ削除」を選択すると、画像は作成者に対して非公開の未分類状態になります。",
|
||||||
"noBoards": "{{boardType}} ボードがありません",
|
"noBoards": "{{boardType}} ボードがありません",
|
||||||
"viewBoards": "ボードを表示",
|
|
||||||
"uncategorizedImages": "分類されていない画像",
|
"uncategorizedImages": "分類されていない画像",
|
||||||
"deleteAllUncategorizedImages": "分類されていないすべての画像を削除",
|
"deleteAllUncategorizedImages": "分類されていないすべての画像を削除"
|
||||||
"deletedImagesCannotBeRestored": "削除した画像は復元できません."
|
|
||||||
},
|
},
|
||||||
"invocationCache": {
|
"invocationCache": {
|
||||||
"invocationCache": "呼び出しキャッシュ",
|
"invocationCache": "呼び出しキャッシュ",
|
||||||
@@ -1757,9 +1685,7 @@
|
|||||||
"strength": "高解像修復の強度",
|
"strength": "高解像修復の強度",
|
||||||
"enabled": "高解像修復が有効"
|
"enabled": "高解像修復が有効"
|
||||||
},
|
},
|
||||||
"enableHrf": "高解像修復を有効",
|
"hrf": "高解像修復"
|
||||||
"hrf": "高解像修復",
|
|
||||||
"upscaleMethod": "アップスケール手法"
|
|
||||||
},
|
},
|
||||||
"prompt": {
|
"prompt": {
|
||||||
"addPromptTrigger": "プロンプトトリガーを追加",
|
"addPromptTrigger": "プロンプトトリガーを追加",
|
||||||
@@ -1769,10 +1695,7 @@
|
|||||||
"expandCurrentPrompt": "現在のプロンプトを展開",
|
"expandCurrentPrompt": "現在のプロンプトを展開",
|
||||||
"uploadImageForPromptGeneration": "プロンプト生成用の画像をアップロードする",
|
"uploadImageForPromptGeneration": "プロンプト生成用の画像をアップロードする",
|
||||||
"expandingPrompt": "プロンプトを展開しています...",
|
"expandingPrompt": "プロンプトを展開しています...",
|
||||||
"resultTitle": "プロンプト拡張完了",
|
|
||||||
"resultSubtitle": "拡張プロンプトの処理方法を選択します:",
|
|
||||||
"replace": "交換する",
|
"replace": "交換する",
|
||||||
"insert": "挿入する",
|
|
||||||
"discard": "破棄する"
|
"discard": "破棄する"
|
||||||
},
|
},
|
||||||
"ui": {
|
"ui": {
|
||||||
@@ -1838,11 +1761,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"controlLayers": {
|
"controlLayers": {
|
||||||
"globalReferenceImage_withCount_other": "全域参照画像",
|
|
||||||
"regionalReferenceImage": "領域参照画像",
|
"regionalReferenceImage": "領域参照画像",
|
||||||
"saveLayerToAssets": "レイヤーをアセットに保存",
|
"saveLayerToAssets": "レイヤーをアセットに保存",
|
||||||
"global": "全域",
|
"global": "全域",
|
||||||
"inpaintMasks_withCount_hidden": "インペイントマスク ({{count}} hidden)",
|
|
||||||
"opacity": "透明度",
|
"opacity": "透明度",
|
||||||
"canvasContextMenu": {
|
"canvasContextMenu": {
|
||||||
"newRegionalGuidance": "新規領域ガイダンス",
|
"newRegionalGuidance": "新規領域ガイダンス",
|
||||||
@@ -1894,7 +1815,6 @@
|
|||||||
"duplicate": "複製",
|
"duplicate": "複製",
|
||||||
"addLayer": "レイヤーを追加",
|
"addLayer": "レイヤーを追加",
|
||||||
"rasterLayer": "ラスターレイヤー",
|
"rasterLayer": "ラスターレイヤー",
|
||||||
"inpaintMasks_withCount_visible": "({{count}}) インペイントマスク",
|
|
||||||
"regional": "領域",
|
"regional": "領域",
|
||||||
"rectangle": "矩形",
|
"rectangle": "矩形",
|
||||||
"moveBackward": "背面へ移動",
|
"moveBackward": "背面へ移動",
|
||||||
@@ -2096,7 +2016,6 @@
|
|||||||
"autoNegative": "オートネガティブ",
|
"autoNegative": "オートネガティブ",
|
||||||
"enableAutoNegative": "オートネガティブを有効にする",
|
"enableAutoNegative": "オートネガティブを有効にする",
|
||||||
"disableAutoNegative": "オートネガティブを無効にする",
|
"disableAutoNegative": "オートネガティブを無効にする",
|
||||||
"deletePrompt": "プロンプトを削除",
|
|
||||||
"deleteReferenceImage": "参照画像を削除",
|
"deleteReferenceImage": "参照画像を削除",
|
||||||
"showHUD": "HUDを表示",
|
"showHUD": "HUDを表示",
|
||||||
"maskFill": "マスク塗りつぶし",
|
"maskFill": "マスク塗りつぶし",
|
||||||
@@ -2108,41 +2027,22 @@
|
|||||||
"addControlLayer": "$t(controlLayers.controlLayer)を追加します",
|
"addControlLayer": "$t(controlLayers.controlLayer)を追加します",
|
||||||
"addInpaintMask": "$t(controlLayers.inpaintMask)を追加します",
|
"addInpaintMask": "$t(controlLayers.inpaintMask)を追加します",
|
||||||
"addRegionalGuidance": "$t(controlLayers.regionalGuidance)を追加します",
|
"addRegionalGuidance": "$t(controlLayers.regionalGuidance)を追加します",
|
||||||
"addGlobalReferenceImage": "$t(controlLayers.globalReferenceImage)を追加します",
|
|
||||||
"addDenoiseLimit": "$t(controlLayers.denoiseLimit)を追加します",
|
"addDenoiseLimit": "$t(controlLayers.denoiseLimit)を追加します",
|
||||||
"controlLayer": "コントロールレイヤー",
|
"controlLayer": "コントロールレイヤー",
|
||||||
"inpaintMask": "インペイントマスク",
|
"inpaintMask": "インペイントマスク",
|
||||||
"referenceImageRegional": "参考画像(地域別)",
|
"referenceImageRegional": "参考画像(地域別)",
|
||||||
"referenceImageGlobal": "参考画像(グローバル)",
|
|
||||||
"asRasterLayer": "$t(controlLayers.rasterLayer) として",
|
"asRasterLayer": "$t(controlLayers.rasterLayer) として",
|
||||||
"asRasterLayerResize": "$t(controlLayers.rasterLayer) として (リサイズ)",
|
"asRasterLayerResize": "$t(controlLayers.rasterLayer) として (リサイズ)",
|
||||||
"asControlLayer": "$t(controlLayers.controlLayer) として",
|
"asControlLayer": "$t(controlLayers.controlLayer) として",
|
||||||
"asControlLayerResize": "$t(controlLayers.controlLayer) として (リサイズ)",
|
"asControlLayerResize": "$t(controlLayers.controlLayer) として (リサイズ)",
|
||||||
"referenceImage": "参照画像",
|
"referenceImage": "参照画像",
|
||||||
"sendingToCanvas": "キャンバスに生成をのせる",
|
|
||||||
"sendingToGallery": "生成をギャラリーに送る",
|
|
||||||
"sendToGallery": "ギャラリーに送る",
|
|
||||||
"sendToGalleryDesc": "Invokeを押すとユニークな画像が生成され、ギャラリーに保存されます。",
|
|
||||||
"sendToCanvas": "キャンバスに送る",
|
"sendToCanvas": "キャンバスに送る",
|
||||||
"newLayerFromImage": "画像から新規レイヤー",
|
"newLayerFromImage": "画像から新規レイヤー",
|
||||||
"newCanvasFromImage": "画像から新規キャンバス",
|
"newCanvasFromImage": "画像から新規キャンバス",
|
||||||
"newImg2ImgCanvasFromImage": "画像からの新規 Img2Img",
|
|
||||||
"copyToClipboard": "クリップボードにコピー",
|
"copyToClipboard": "クリップボードにコピー",
|
||||||
"sendToCanvasDesc": "Invokeを押すと、進行中の作品がキャンバス上にステージされます。",
|
|
||||||
"viewProgressInViewer": "<Btn>画像ビューア</Btn>で進行状況と出力を表示します。",
|
|
||||||
"viewProgressOnCanvas": "<Btn>キャンバス</Btn> で進行状況とステージ出力を表示します。",
|
|
||||||
"rasterLayer_withCount_other": "ラスターレイヤー",
|
"rasterLayer_withCount_other": "ラスターレイヤー",
|
||||||
"controlLayer_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_other": "レイヤー",
|
||||||
"layer_withCount_other": "レイヤー ({{count}})",
|
|
||||||
"convertRasterLayerTo": "$t(controlLayers.rasterLayer) を変換する",
|
"convertRasterLayerTo": "$t(controlLayers.rasterLayer) を変換する",
|
||||||
"convertControlLayerTo": "$t(controlLayers.controlLayer) を変換する",
|
"convertControlLayerTo": "$t(controlLayers.controlLayer) を変換する",
|
||||||
"convertRegionalGuidanceTo": "$t(controlLayers.regionalGuidance) を変換する",
|
"convertRegionalGuidanceTo": "$t(controlLayers.regionalGuidance) を変換する",
|
||||||
@@ -2160,7 +2060,6 @@
|
|||||||
"pasteToBboxDesc": "新しいレイヤー(Bbox内)",
|
"pasteToBboxDesc": "新しいレイヤー(Bbox内)",
|
||||||
"pasteToCanvas": "キャンバス",
|
"pasteToCanvas": "キャンバス",
|
||||||
"pasteToCanvasDesc": "新しいレイヤー(キャンバス内)",
|
"pasteToCanvasDesc": "新しいレイヤー(キャンバス内)",
|
||||||
"pastedTo": "{{destination}} に貼り付けました",
|
|
||||||
"transparency": "透明性",
|
"transparency": "透明性",
|
||||||
"enableTransparencyEffect": "透明効果を有効にする",
|
"enableTransparencyEffect": "透明効果を有効にする",
|
||||||
"disableTransparencyEffect": "透明効果を無効にする",
|
"disableTransparencyEffect": "透明効果を無効にする",
|
||||||
@@ -2173,7 +2072,6 @@
|
|||||||
"locked": "ロックされています",
|
"locked": "ロックされています",
|
||||||
"unlocked": "ロック解除",
|
"unlocked": "ロック解除",
|
||||||
"deleteSelected": "選択項目を削除",
|
"deleteSelected": "選択項目を削除",
|
||||||
"stagingOnCanvas": "ステージング画像",
|
|
||||||
"replaceLayer": "レイヤーの置き換え",
|
"replaceLayer": "レイヤーの置き換え",
|
||||||
"pullBboxIntoLayer": "Bboxをレイヤーに引き込む",
|
"pullBboxIntoLayer": "Bboxをレイヤーに引き込む",
|
||||||
"pullBboxIntoReferenceImage": "Bboxを参照画像に取り込む",
|
"pullBboxIntoReferenceImage": "Bboxを参照画像に取り込む",
|
||||||
@@ -2181,17 +2079,11 @@
|
|||||||
"useImage": "画像を使う",
|
"useImage": "画像を使う",
|
||||||
"negativePrompt": "ネガティブプロンプト",
|
"negativePrompt": "ネガティブプロンプト",
|
||||||
"beginEndStepPercentShort": "開始/終了 %",
|
"beginEndStepPercentShort": "開始/終了 %",
|
||||||
"newGallerySession": "新しいギャラリーセッション",
|
|
||||||
"newGallerySessionDesc": "これにより、キャンバスとモデル選択以外のすべての設定がクリアされます。生成した画像はギャラリーに送信されます。",
|
|
||||||
"newCanvasSession": "新規キャンバスセッション",
|
|
||||||
"newCanvasSessionDesc": "これにより、キャンバスとモデル選択以外のすべての設定がクリアされます。生成はキャンバス上でステージングされます。",
|
|
||||||
"resetCanvasLayers": "キャンバスレイヤーをリセット",
|
"resetCanvasLayers": "キャンバスレイヤーをリセット",
|
||||||
"resetGenerationSettings": "生成設定をリセット",
|
"resetGenerationSettings": "生成設定をリセット",
|
||||||
"replaceCurrent": "現在のものを置き換える",
|
|
||||||
"controlLayerEmptyState": "<UploadButton>画像をアップロード</UploadButton>、<GalleryButton>ギャラリー</GalleryButton>からこのレイヤーに画像をドラッグ、<PullBboxButton>境界ボックスをこのレイヤーにプル</PullBboxButton>、またはキャンバスに描画して開始します。",
|
"controlLayerEmptyState": "<UploadButton>画像をアップロード</UploadButton>、<GalleryButton>ギャラリー</GalleryButton>からこのレイヤーに画像をドラッグ、<PullBboxButton>境界ボックスをこのレイヤーにプル</PullBboxButton>、またはキャンバスに描画して開始します。",
|
||||||
"referenceImageEmptyStateWithCanvasOptions": "開始するには、<UploadButton>画像をアップロード</UploadButton>するか、<GalleryButton>ギャラリー</GalleryButton>からこの参照画像に画像をドラッグするか、<PullBboxButton>境界ボックスをこの参照画像にプル</PullBboxButton>します。",
|
"referenceImageEmptyStateWithCanvasOptions": "開始するには、<UploadButton>画像をアップロード</UploadButton>するか、<GalleryButton>ギャラリー</GalleryButton>からこの参照画像に画像をドラッグするか、<PullBboxButton>境界ボックスをこの参照画像にプル</PullBboxButton>します。",
|
||||||
"referenceImageEmptyState": "開始するには、<UploadButton>画像をアップロード</UploadButton>するか、<GalleryButton>ギャラリー</GalleryButton>からこの参照画像に画像をドラッグします。",
|
"referenceImageEmptyState": "開始するには、<UploadButton>画像をアップロード</UploadButton>するか、<GalleryButton>ギャラリー</GalleryButton>からこの参照画像に画像をドラッグします。",
|
||||||
"uploadOrDragAnImage": "ギャラリーから画像をドラッグするか、<UploadButton>画像をアップロード</UploadButton>します。",
|
|
||||||
"imageNoise": "画像ノイズ",
|
"imageNoise": "画像ノイズ",
|
||||||
"denoiseLimit": "ノイズ除去制限",
|
"denoiseLimit": "ノイズ除去制限",
|
||||||
"warnings": {
|
"warnings": {
|
||||||
@@ -2257,9 +2149,6 @@
|
|||||||
"saveAs": "名前を付けて保存",
|
"saveAs": "名前を付けて保存",
|
||||||
"cancel": "キャンセル",
|
"cancel": "キャンセル",
|
||||||
"process": "プロセス",
|
"process": "プロセス",
|
||||||
"help1": "ターゲットオブジェクトを1つ選択します。<Bold>含める</Bold>ポイントと<Bold>除外</Bold>ポイントを追加して、レイヤーのどの部分がターゲットオブジェクトの一部であるかを示します。",
|
|
||||||
"help2": "対象オブジェクト内に<Bold>含める</Bold>ポイントを1つ選択するところから始めます。ポイントを追加して選択範囲を絞り込みます。ポイントが少ないほど、通常はより良い結果が得られます。",
|
|
||||||
"help3": "選択を反転して、ターゲットオブジェクト以外のすべてを選択します。",
|
|
||||||
"clickToAdd": "レイヤーをクリックしてポイントを追加します",
|
"clickToAdd": "レイヤーをクリックしてポイントを追加します",
|
||||||
"dragToMove": "ポイントをドラッグして移動します",
|
"dragToMove": "ポイントをドラッグして移動します",
|
||||||
"clickToRemove": "ポイントをクリックして削除します"
|
"clickToRemove": "ポイントをクリックして削除します"
|
||||||
@@ -2360,12 +2249,8 @@
|
|||||||
"loading": "ロード中...",
|
"loading": "ロード中...",
|
||||||
"steps": "ステップ",
|
"steps": "ステップ",
|
||||||
"refiner": "Refiner",
|
"refiner": "Refiner",
|
||||||
"negStylePrompt": "ネガティブスタイルプロンプト",
|
|
||||||
"noModelsAvailable": "利用できるモデルがありません",
|
"noModelsAvailable": "利用できるモデルがありません",
|
||||||
"posStylePrompt": "ポジティブスタイルプロンプト",
|
|
||||||
"cfgScale": "CFGスケール",
|
"cfgScale": "CFGスケール",
|
||||||
"concatPromptStyle": "リンキングプロンプトとスタイル",
|
|
||||||
"freePromptStyle": "手動スタイルプロンプト",
|
|
||||||
"posAestheticScore": "ポジティブ美的スコア",
|
"posAestheticScore": "ポジティブ美的スコア",
|
||||||
"refinerSteps": "リファイナーステップ",
|
"refinerSteps": "リファイナーステップ",
|
||||||
"refinerStart": "リファイナースタート",
|
"refinerStart": "リファイナースタート",
|
||||||
@@ -2383,8 +2268,6 @@
|
|||||||
"name": "名前",
|
"name": "名前",
|
||||||
"descending": "降順",
|
"descending": "降順",
|
||||||
"searchPlaceholder": "名前、説明、タグで検索",
|
"searchPlaceholder": "名前、説明、タグで検索",
|
||||||
"projectWorkflows": "プロジェクトワークフロー",
|
|
||||||
"searchWorkflows": "ワークフローを検索",
|
|
||||||
"updated": "アップデート",
|
"updated": "アップデート",
|
||||||
"published": "公表",
|
"published": "公表",
|
||||||
"builder": {
|
"builder": {
|
||||||
@@ -2410,10 +2293,8 @@
|
|||||||
"addToForm": "フォームに追加",
|
"addToForm": "フォームに追加",
|
||||||
"headingPlaceholder": "空の見出し",
|
"headingPlaceholder": "空の見出し",
|
||||||
"nodeFieldTooltip": "ノード フィールドを追加するには、ワークフロー エディターのフィールドにある小さなプラス記号ボタンをクリックするか、フィールド名をフォームにドラッグします。",
|
"nodeFieldTooltip": "ノード フィールドを追加するには、ワークフロー エディターのフィールドにある小さなプラス記号ボタンをクリックするか、フィールド名をフォームにドラッグします。",
|
||||||
"workflowBuilderAlphaWarning": "ワークフロービルダーは現在アルファ版です。安定版リリースまでに互換性に影響する変更が発生する可能性があります。",
|
|
||||||
"component": "コンポーネント",
|
"component": "コンポーネント",
|
||||||
"textPlaceholder": "空のテキスト",
|
"textPlaceholder": "空のテキスト",
|
||||||
"emptyRootPlaceholderViewMode": "このワークフローのフォームの作成を開始するには、[編集] をクリックします。",
|
|
||||||
"addOption": "オプションを追加",
|
"addOption": "オプションを追加",
|
||||||
"singleLine": "単線",
|
"singleLine": "単線",
|
||||||
"numberInput": "数値入力",
|
"numberInput": "数値入力",
|
||||||
@@ -2464,20 +2345,15 @@
|
|||||||
"convertGraph": "グラフを変換",
|
"convertGraph": "グラフを変換",
|
||||||
"downloadWorkflow": "ファイルに保存",
|
"downloadWorkflow": "ファイルに保存",
|
||||||
"saveWorkflow": "ワークフローを保存",
|
"saveWorkflow": "ワークフローを保存",
|
||||||
"userWorkflows": "ユーザーワークフロー",
|
|
||||||
"yourWorkflows": "あなたのワークフロー",
|
"yourWorkflows": "あなたのワークフロー",
|
||||||
"edit": "編集",
|
"edit": "編集",
|
||||||
"workflowLibrary": "ワークフローライブラリ",
|
"workflowLibrary": "ワークフローライブラリ",
|
||||||
"workflowSaved": "ワークフローが保存されました",
|
"workflowSaved": "ワークフローが保存されました",
|
||||||
"clearWorkflowSearchFilter": "ワークフロー検索フィルタをクリア",
|
|
||||||
"workflowCleared": "ワークフローが作成されました",
|
"workflowCleared": "ワークフローが作成されました",
|
||||||
"autoLayout": "オートレイアウト",
|
"autoLayout": "オートレイアウト",
|
||||||
"view": "ビュー",
|
"view": "ビュー",
|
||||||
"saveChanges": "変更を保存",
|
"saveChanges": "変更を保存",
|
||||||
"noDescription": "説明なし",
|
|
||||||
"recommended": "あなたへのおすすめ",
|
"recommended": "あなたへのおすすめ",
|
||||||
"noRecentWorkflows": "最近のワークフローがありません",
|
|
||||||
"problemLoading": "ワークフローのローディングに関する問題",
|
|
||||||
"newWorkflowCreated": "新しいワークフローが作成されました",
|
"newWorkflowCreated": "新しいワークフローが作成されました",
|
||||||
"noWorkflows": "ワークフローがありません",
|
"noWorkflows": "ワークフローがありません",
|
||||||
"copyShareLink": "共有リンクをコピー",
|
"copyShareLink": "共有リンクをコピー",
|
||||||
@@ -2485,21 +2361,16 @@
|
|||||||
"workflowThumbnail": "ワークフローサムネイル",
|
"workflowThumbnail": "ワークフローサムネイル",
|
||||||
"loadWorkflow": "$t(common.load) ワークフロー",
|
"loadWorkflow": "$t(common.load) ワークフロー",
|
||||||
"shared": "共有",
|
"shared": "共有",
|
||||||
"openWorkflow": "ワークフローを開く",
|
|
||||||
"emptyStringPlaceholder": "<空の文字列>",
|
"emptyStringPlaceholder": "<空の文字列>",
|
||||||
"browseWorkflows": "ワークフローを閲覧する",
|
"browseWorkflows": "ワークフローを閲覧する",
|
||||||
"saveWorkflowAs": "ワークフローとして保存",
|
"saveWorkflowAs": "ワークフローとして保存",
|
||||||
"private": "プライベート",
|
"private": "プライベート",
|
||||||
"deselectAll": "すべて選択解除",
|
"deselectAll": "すべて選択解除",
|
||||||
"delete": "削除",
|
"delete": "削除",
|
||||||
"openLibrary": "ライブラリを開く",
|
|
||||||
"loadMore": "もっと読み込む",
|
"loadMore": "もっと読み込む",
|
||||||
"saveWorkflowToProject": "ワークフローをプロジェクトに保存",
|
"saveWorkflowToProject": "ワークフローをプロジェクトに保存",
|
||||||
"created": "作成されました",
|
"created": "作成されました",
|
||||||
"workflowEditorMenu": "ワークフローエディターメニュー",
|
"workflowEditorMenu": "ワークフローエディターメニュー",
|
||||||
"defaultWorkflows": "デフォルトワークフロー",
|
|
||||||
"allLoaded": "すべてのワークフローが読み込まれました",
|
|
||||||
"filterByTags": "タグでフィルター",
|
|
||||||
"recentlyOpened": "最近開いた",
|
"recentlyOpened": "最近開いた",
|
||||||
"opened": "オープン",
|
"opened": "オープン",
|
||||||
"deleteWorkflow": "ワークフローを削除",
|
"deleteWorkflow": "ワークフローを削除",
|
||||||
@@ -2545,7 +2416,6 @@
|
|||||||
"perIterationDesc": "それぞれのいてレーションに別のシードを使う"
|
"perIterationDesc": "それぞれのいてレーションに別のシードを使う"
|
||||||
},
|
},
|
||||||
"showDynamicPrompts": "ダイナミックプロンプトを表示する",
|
"showDynamicPrompts": "ダイナミックプロンプトを表示する",
|
||||||
"promptsToGenerate": "生成するプロンプト",
|
|
||||||
"dynamicPrompts": "ダイナミックプロンプト",
|
"dynamicPrompts": "ダイナミックプロンプト",
|
||||||
"loading": "ダイナミックプロンプトを生成...",
|
"loading": "ダイナミックプロンプトを生成...",
|
||||||
"maxPrompts": "最大プロンプト"
|
"maxPrompts": "最大プロンプト"
|
||||||
@@ -2571,8 +2441,7 @@
|
|||||||
"キャンバス: SDXL のアスペクト比がスマートになり、スクロールによるズームが改善されました。"
|
"キャンバス: SDXL のアスペクト比がスマートになり、スクロールによるズームが改善されました。"
|
||||||
],
|
],
|
||||||
"readReleaseNotes": "リリースノートを読む",
|
"readReleaseNotes": "リリースノートを読む",
|
||||||
"watchRecentReleaseVideos": "最近のリリースビデオを見る",
|
"watchRecentReleaseVideos": "最近のリリースビデオを見る"
|
||||||
"watchUiUpdatesOverview": "Watch UI アップデートの概要"
|
|
||||||
},
|
},
|
||||||
"supportVideos": {
|
"supportVideos": {
|
||||||
"supportVideos": "サポートビデオ",
|
"supportVideos": "サポートビデオ",
|
||||||
|
|||||||
@@ -27,7 +27,6 @@
|
|||||||
"save": "저장",
|
"save": "저장",
|
||||||
"created": "생성됨",
|
"created": "생성됨",
|
||||||
"error": "에러",
|
"error": "에러",
|
||||||
"prevPage": "이전 페이지",
|
|
||||||
"ipAdapter": "IP 어댑터",
|
"ipAdapter": "IP 어댑터",
|
||||||
"installed": "설치됨",
|
"installed": "설치됨",
|
||||||
"accept": "수락",
|
"accept": "수락",
|
||||||
@@ -42,7 +41,6 @@
|
|||||||
"outputs": "결과물",
|
"outputs": "결과물",
|
||||||
"unknownError": "알려지지 않은 에러",
|
"unknownError": "알려지지 않은 에러",
|
||||||
"linear": "선형",
|
"linear": "선형",
|
||||||
"imageFailedToLoad": "이미지를 로드할 수 없음",
|
|
||||||
"direction": "방향",
|
"direction": "방향",
|
||||||
"data": "데이터",
|
"data": "데이터",
|
||||||
"somethingWentWrong": "뭔가 잘못됐어요",
|
"somethingWentWrong": "뭔가 잘못됐어요",
|
||||||
@@ -52,7 +50,6 @@
|
|||||||
"orderBy": "정렬 기준",
|
"orderBy": "정렬 기준",
|
||||||
"copyError": "$t(gallery.copy) 에러",
|
"copyError": "$t(gallery.copy) 에러",
|
||||||
"learnMore": "더 알아보기",
|
"learnMore": "더 알아보기",
|
||||||
"nextPage": "다음 페이지",
|
|
||||||
"saveAs": "다른 이름으로 저장",
|
"saveAs": "다른 이름으로 저장",
|
||||||
"loading": "불러오는 중",
|
"loading": "불러오는 중",
|
||||||
"random": "랜덤",
|
"random": "랜덤",
|
||||||
@@ -60,18 +57,15 @@
|
|||||||
"postprocessing": "후처리",
|
"postprocessing": "후처리",
|
||||||
"advanced": "고급",
|
"advanced": "고급",
|
||||||
"input": "입력",
|
"input": "입력",
|
||||||
"details": "세부사항",
|
"details": "세부사항"
|
||||||
"notInstalled": "설치되지 않음"
|
|
||||||
},
|
},
|
||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "이미지 크기",
|
"galleryImageSize": "이미지 크기",
|
||||||
"gallerySettings": "갤러리 설정",
|
"gallerySettings": "갤러리 설정",
|
||||||
"deleteSelection": "선택 항목 삭제",
|
"deleteSelection": "선택 항목 삭제",
|
||||||
"featuresWillReset": "이 이미지를 삭제하면 해당 기능이 즉시 재설정됩니다.",
|
"featuresWillReset": "이 이미지를 삭제하면 해당 기능이 즉시 재설정됩니다.",
|
||||||
"noImagesInGallery": "보여줄 이미지가 없음",
|
|
||||||
"autoSwitchNewImages": "새로운 이미지로 자동 전환",
|
"autoSwitchNewImages": "새로운 이미지로 자동 전환",
|
||||||
"loading": "불러오는 중",
|
"loading": "불러오는 중",
|
||||||
"unableToLoad": "갤러리를 로드할 수 없음",
|
|
||||||
"image": "이미지",
|
"image": "이미지",
|
||||||
"drop": "드랍",
|
"drop": "드랍",
|
||||||
"downloadSelection": "선택 항목 다운로드",
|
"downloadSelection": "선택 항목 다운로드",
|
||||||
@@ -151,8 +145,6 @@
|
|||||||
"loadWorkflow": "Workflow 불러오기",
|
"loadWorkflow": "Workflow 불러오기",
|
||||||
"noOutputRecorded": "기록된 출력 없음",
|
"noOutputRecorded": "기록된 출력 없음",
|
||||||
"colorCodeEdgesHelp": "연결된 필드에 따른 색상 코드 선",
|
"colorCodeEdgesHelp": "연결된 필드에 따른 색상 코드 선",
|
||||||
"hideLegendNodes": "필드 유형 범례 숨기기",
|
|
||||||
"addLinearView": "Linear View에 추가",
|
|
||||||
"float": "실수",
|
"float": "실수",
|
||||||
"targetNodeFieldDoesNotExist": "잘못된 모서리: 대상/입력 필드 {{node}}. {{field}}이(가) 없습니다",
|
"targetNodeFieldDoesNotExist": "잘못된 모서리: 대상/입력 필드 {{node}}. {{field}}이(가) 없습니다",
|
||||||
"animatedEdges": "애니메이션 모서리",
|
"animatedEdges": "애니메이션 모서리",
|
||||||
@@ -160,7 +152,6 @@
|
|||||||
"nodeTemplate": "노드 템플릿",
|
"nodeTemplate": "노드 템플릿",
|
||||||
"nodeOpacity": "노드 불투명도",
|
"nodeOpacity": "노드 불투명도",
|
||||||
"sourceNodeDoesNotExist": "잘못된 모서리: 소스/출력 노드 {{node}}이(가) 없습니다",
|
"sourceNodeDoesNotExist": "잘못된 모서리: 소스/출력 노드 {{node}}이(가) 없습니다",
|
||||||
"noFieldsLinearview": "Linear View에 추가된 필드 없음",
|
|
||||||
"nodeSearch": "노드 검색",
|
"nodeSearch": "노드 검색",
|
||||||
"inputMayOnlyHaveOneConnection": "입력에 하나의 연결만 있을 수 있습니다",
|
"inputMayOnlyHaveOneConnection": "입력에 하나의 연결만 있을 수 있습니다",
|
||||||
"notes": "메모",
|
"notes": "메모",
|
||||||
@@ -195,7 +186,6 @@
|
|||||||
"notesDescription": "Workflow에 대한 메모 추가",
|
"notesDescription": "Workflow에 대한 메모 추가",
|
||||||
"colorCodeEdges": "색상-코드 선",
|
"colorCodeEdges": "색상-코드 선",
|
||||||
"targetNodeDoesNotExist": "잘못된 모서리: 대상/입력 노드 {{node}}이(가) 없습니다",
|
"targetNodeDoesNotExist": "잘못된 모서리: 대상/입력 노드 {{node}}이(가) 없습니다",
|
||||||
"mismatchedVersion": "잘못된 노드: {{type}} 유형의 {{node}} 노드에 일치하지 않는 버전이 있습니다(업데이트 해보시겠습니까?)",
|
|
||||||
"addNodeToolTip": "노드 추가(Shift+A, Space)",
|
"addNodeToolTip": "노드 추가(Shift+A, Space)",
|
||||||
"collectionOrScalarFieldType": "{{name}} 컬렉션|Scalar",
|
"collectionOrScalarFieldType": "{{name}} 컬렉션|Scalar",
|
||||||
"nodeVersion": "노드 버전",
|
"nodeVersion": "노드 버전",
|
||||||
@@ -242,7 +232,6 @@
|
|||||||
"next": "다음",
|
"next": "다음",
|
||||||
"cancelBatch": "Batch 취소",
|
"cancelBatch": "Batch 취소",
|
||||||
"back": "back",
|
"back": "back",
|
||||||
"batchFieldValues": "Batch 필드 값들",
|
|
||||||
"cancel": "취소",
|
"cancel": "취소",
|
||||||
"session": "세션",
|
"session": "세션",
|
||||||
"time": "시간",
|
"time": "시간",
|
||||||
@@ -296,8 +285,6 @@
|
|||||||
"cacheSize": "캐시 크기"
|
"cacheSize": "캐시 크기"
|
||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"enableHrf": "이용 가능한 고해상도 고정",
|
|
||||||
"upscaleMethod": "업스케일 방법",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"strength": "고해상도 고정 강도",
|
"strength": "고해상도 고정 강도",
|
||||||
"enabled": "고해상도 고정 사용",
|
"enabled": "고해상도 고정 사용",
|
||||||
@@ -308,12 +295,10 @@
|
|||||||
"models": {
|
"models": {
|
||||||
"noMatchingModels": "일치하는 모델 없음",
|
"noMatchingModels": "일치하는 모델 없음",
|
||||||
"loading": "로딩중",
|
"loading": "로딩중",
|
||||||
"noMatchingLoRAs": "일치하는 LoRA 없음",
|
|
||||||
"noModelsAvailable": "사용 가능한 모델이 없음",
|
"noModelsAvailable": "사용 가능한 모델이 없음",
|
||||||
"addLora": "LoRA 추가",
|
"addLora": "LoRA 추가",
|
||||||
"selectModel": "모델 선택",
|
"selectModel": "모델 선택",
|
||||||
"noRefinerModelsInstalled": "SDXL Refiner 모델이 설치되지 않음",
|
"noRefinerModelsInstalled": "SDXL Refiner 모델이 설치되지 않음"
|
||||||
"noLoRAsInstalled": "설치된 LoRA 없음"
|
|
||||||
},
|
},
|
||||||
"boards": {
|
"boards": {
|
||||||
"autoAddBoard": "자동 추가 Board",
|
"autoAddBoard": "자동 추가 Board",
|
||||||
|
|||||||
@@ -30,12 +30,10 @@
|
|||||||
"ipAdapter": "IP-adapter",
|
"ipAdapter": "IP-adapter",
|
||||||
"auto": "Autom.",
|
"auto": "Autom.",
|
||||||
"controlNet": "ControlNet",
|
"controlNet": "ControlNet",
|
||||||
"imageFailedToLoad": "Kan afbeelding niet laden",
|
|
||||||
"learnMore": "Meer informatie",
|
"learnMore": "Meer informatie",
|
||||||
"advanced": "Uitgebreid",
|
"advanced": "Uitgebreid",
|
||||||
"file": "Bestand",
|
"file": "Bestand",
|
||||||
"installed": "Geïnstalleerd",
|
"installed": "Geïnstalleerd",
|
||||||
"notInstalled": "Niet $t(common.installed)",
|
|
||||||
"simple": "Eenvoudig",
|
"simple": "Eenvoudig",
|
||||||
"somethingWentWrong": "Er ging iets mis",
|
"somethingWentWrong": "Er ging iets mis",
|
||||||
"add": "Voeg toe",
|
"add": "Voeg toe",
|
||||||
@@ -43,14 +41,12 @@
|
|||||||
"details": "Details",
|
"details": "Details",
|
||||||
"outputs": "Uitvoeren",
|
"outputs": "Uitvoeren",
|
||||||
"save": "Bewaar",
|
"save": "Bewaar",
|
||||||
"nextPage": "Volgende pagina",
|
|
||||||
"blue": "Blauw",
|
"blue": "Blauw",
|
||||||
"alpha": "Alfa",
|
"alpha": "Alfa",
|
||||||
"red": "Rood",
|
"red": "Rood",
|
||||||
"editor": "Editor",
|
"editor": "Editor",
|
||||||
"folder": "Map",
|
"folder": "Map",
|
||||||
"format": "structuur",
|
"format": "structuur",
|
||||||
"goTo": "Ga naar",
|
|
||||||
"template": "Sjabloon",
|
"template": "Sjabloon",
|
||||||
"input": "Invoer",
|
"input": "Invoer",
|
||||||
"safetensors": "Safetensors",
|
"safetensors": "Safetensors",
|
||||||
@@ -62,7 +58,6 @@
|
|||||||
"negativePrompt": "Negatieve prompt",
|
"negativePrompt": "Negatieve prompt",
|
||||||
"selected": "Geselecteerd",
|
"selected": "Geselecteerd",
|
||||||
"orderBy": "Sorteer op",
|
"orderBy": "Sorteer op",
|
||||||
"prevPage": "Vorige pagina",
|
|
||||||
"beta": "Bèta",
|
"beta": "Bèta",
|
||||||
"copyError": "$t(gallery.copy) Fout",
|
"copyError": "$t(gallery.copy) Fout",
|
||||||
"toResolve": "Op te lossen",
|
"toResolve": "Op te lossen",
|
||||||
@@ -79,21 +74,18 @@
|
|||||||
"delete": "Verwijder",
|
"delete": "Verwijder",
|
||||||
"direction": "Richting",
|
"direction": "Richting",
|
||||||
"error": "Fout",
|
"error": "Fout",
|
||||||
"localSystem": "Lokaal systeem",
|
|
||||||
"unknownError": "Onbekende fout"
|
"unknownError": "Onbekende fout"
|
||||||
},
|
},
|
||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "Afbeeldingsgrootte",
|
"galleryImageSize": "Afbeeldingsgrootte",
|
||||||
"gallerySettings": "Instellingen galerij",
|
"gallerySettings": "Instellingen galerij",
|
||||||
"autoSwitchNewImages": "Wissel autom. naar nieuwe afbeeldingen",
|
"autoSwitchNewImages": "Wissel autom. naar nieuwe afbeeldingen",
|
||||||
"noImagesInGallery": "Geen afbeeldingen om te tonen",
|
|
||||||
"deleteImage_one": "Verwijder afbeelding",
|
"deleteImage_one": "Verwijder afbeelding",
|
||||||
"deleteImage_other": "",
|
"deleteImage_other": "",
|
||||||
"deleteImagePermanent": "Verwijderde afbeeldingen kunnen niet worden hersteld.",
|
"deleteImagePermanent": "Verwijderde afbeeldingen kunnen niet worden hersteld.",
|
||||||
"autoAssignBoardOnClick": "Ken automatisch bord toe bij klikken",
|
"autoAssignBoardOnClick": "Ken automatisch bord toe bij klikken",
|
||||||
"featuresWillReset": "Als je deze afbeelding verwijdert, dan worden deze functies onmiddellijk teruggezet.",
|
"featuresWillReset": "Als je deze afbeelding verwijdert, dan worden deze functies onmiddellijk teruggezet.",
|
||||||
"loading": "Bezig met laden",
|
"loading": "Bezig met laden",
|
||||||
"unableToLoad": "Kan galerij niet laden",
|
|
||||||
"downloadSelection": "Download selectie",
|
"downloadSelection": "Download selectie",
|
||||||
"currentlyInUse": "Deze afbeelding is momenteel in gebruik door de volgende functies:",
|
"currentlyInUse": "Deze afbeelding is momenteel in gebruik door de volgende functies:",
|
||||||
"copy": "Kopieer",
|
"copy": "Kopieer",
|
||||||
@@ -199,12 +191,10 @@
|
|||||||
"scaledHeight": "Geschaalde H",
|
"scaledHeight": "Geschaalde H",
|
||||||
"infillMethod": "Infill-methode",
|
"infillMethod": "Infill-methode",
|
||||||
"tileSize": "Grootte tegel",
|
"tileSize": "Grootte tegel",
|
||||||
"downloadImage": "Download afbeelding",
|
|
||||||
"usePrompt": "Hergebruik invoertekst",
|
"usePrompt": "Hergebruik invoertekst",
|
||||||
"useSeed": "Hergebruik seed",
|
"useSeed": "Hergebruik seed",
|
||||||
"useAll": "Hergebruik alles",
|
"useAll": "Hergebruik alles",
|
||||||
"info": "Info",
|
"info": "Info",
|
||||||
"showOptionsPanel": "Toon deelscherm Opties (O of T)",
|
|
||||||
"symmetry": "Symmetrie",
|
"symmetry": "Symmetrie",
|
||||||
"cancel": {
|
"cancel": {
|
||||||
"cancel": "Annuleer"
|
"cancel": "Annuleer"
|
||||||
@@ -293,15 +283,12 @@
|
|||||||
"baseModelChangedCleared_one": "Basismodel is gewijzigd: {{count}} niet-compatibel submodel weggehaald of uitgeschakeld",
|
"baseModelChangedCleared_one": "Basismodel is gewijzigd: {{count}} niet-compatibel submodel weggehaald of uitgeschakeld",
|
||||||
"baseModelChangedCleared_other": "Basismodel is gewijzigd: {{count}} niet-compatibele submodellen weggehaald of uitgeschakeld",
|
"baseModelChangedCleared_other": "Basismodel is gewijzigd: {{count}} niet-compatibele submodellen weggehaald of uitgeschakeld",
|
||||||
"loadedWithWarnings": "Werkstroom geladen met waarschuwingen",
|
"loadedWithWarnings": "Werkstroom geladen met waarschuwingen",
|
||||||
"setControlImage": "Ingesteld als controle-afbeelding",
|
|
||||||
"setNodeField": "Ingesteld als knooppuntveld",
|
|
||||||
"imageUploaded": "Afbeelding geüpload",
|
"imageUploaded": "Afbeelding geüpload",
|
||||||
"addedToBoard": "Toegevoegd aan bord",
|
"addedToBoard": "Toegevoegd aan bord",
|
||||||
"workflowLoaded": "Werkstroom geladen",
|
"workflowLoaded": "Werkstroom geladen",
|
||||||
"modelAddedSimple": "Model toegevoegd aan wachtrij",
|
"modelAddedSimple": "Model toegevoegd aan wachtrij",
|
||||||
"imageUploadFailed": "Fout bij uploaden afbeelding",
|
"imageUploadFailed": "Fout bij uploaden afbeelding",
|
||||||
"workflowDeleted": "Werkstroom verwijderd",
|
"workflowDeleted": "Werkstroom verwijderd",
|
||||||
"invalidUpload": "Ongeldige upload",
|
|
||||||
"problemRetrievingWorkflow": "Fout bij ophalen van werkstroom",
|
"problemRetrievingWorkflow": "Fout bij ophalen van werkstroom",
|
||||||
"parameters": "Parameters",
|
"parameters": "Parameters",
|
||||||
"modelImportCanceled": "Importeren model geannuleerd",
|
"modelImportCanceled": "Importeren model geannuleerd",
|
||||||
@@ -325,17 +312,14 @@
|
|||||||
"zoomOutNodes": "Uitzoomen",
|
"zoomOutNodes": "Uitzoomen",
|
||||||
"fitViewportNodes": "Aanpassen aan beeld",
|
"fitViewportNodes": "Aanpassen aan beeld",
|
||||||
"hideMinimapnodes": "Minimap verbergen",
|
"hideMinimapnodes": "Minimap verbergen",
|
||||||
"showLegendNodes": "Typelegende veld tonen",
|
|
||||||
"zoomInNodes": "Inzoomen",
|
"zoomInNodes": "Inzoomen",
|
||||||
"showMinimapnodes": "Minimap tonen",
|
"showMinimapnodes": "Minimap tonen",
|
||||||
"hideLegendNodes": "Typelegende veld verbergen",
|
|
||||||
"reloadNodeTemplates": "Herlaad knooppuntsjablonen",
|
"reloadNodeTemplates": "Herlaad knooppuntsjablonen",
|
||||||
"loadWorkflow": "Laad werkstroom",
|
"loadWorkflow": "Laad werkstroom",
|
||||||
"downloadWorkflow": "Download JSON van werkstroom",
|
"downloadWorkflow": "Download JSON van werkstroom",
|
||||||
"scheduler": "Planner",
|
"scheduler": "Planner",
|
||||||
"missingTemplate": "Ongeldig knooppunt: knooppunt {{node}} van het soort {{type}} heeft een ontbrekend sjabloon (niet geïnstalleerd?)",
|
"missingTemplate": "Ongeldig knooppunt: knooppunt {{node}} van het soort {{type}} heeft een ontbrekend sjabloon (niet geïnstalleerd?)",
|
||||||
"workflowDescription": "Korte beschrijving",
|
"workflowDescription": "Korte beschrijving",
|
||||||
"versionUnknown": " Versie onbekend",
|
|
||||||
"noNodeSelected": "Geen knooppunt gekozen",
|
"noNodeSelected": "Geen knooppunt gekozen",
|
||||||
"addNode": "Voeg knooppunt toe",
|
"addNode": "Voeg knooppunt toe",
|
||||||
"unableToValidateWorkflow": "Kan werkstroom niet valideren",
|
"unableToValidateWorkflow": "Kan werkstroom niet valideren",
|
||||||
@@ -349,9 +333,7 @@
|
|||||||
"integer": "Geheel getal",
|
"integer": "Geheel getal",
|
||||||
"nodeTemplate": "Sjabloon knooppunt",
|
"nodeTemplate": "Sjabloon knooppunt",
|
||||||
"nodeOpacity": "Dekking knooppunt",
|
"nodeOpacity": "Dekking knooppunt",
|
||||||
"unableToLoadWorkflow": "Fout bij laden werkstroom",
|
|
||||||
"snapToGrid": "Lijn uit op raster",
|
"snapToGrid": "Lijn uit op raster",
|
||||||
"noFieldsLinearview": "Geen velden toegevoegd aan lineaire weergave",
|
|
||||||
"nodeSearch": "Zoek naar knooppunten",
|
"nodeSearch": "Zoek naar knooppunten",
|
||||||
"updateNode": "Werk knooppunt bij",
|
"updateNode": "Werk knooppunt bij",
|
||||||
"version": "Versie",
|
"version": "Versie",
|
||||||
@@ -370,9 +352,7 @@
|
|||||||
"edge": "Rand",
|
"edge": "Rand",
|
||||||
"animatedEdgesHelp": "Animeer gekozen randen en randen verbonden met de gekozen knooppunten",
|
"animatedEdgesHelp": "Animeer gekozen randen en randen verbonden met de gekozen knooppunten",
|
||||||
"cannotDuplicateConnection": "Kan geen dubbele verbindingen maken",
|
"cannotDuplicateConnection": "Kan geen dubbele verbindingen maken",
|
||||||
"unknownTemplate": "Onbekend sjabloon",
|
|
||||||
"noWorkflow": "Geen werkstroom",
|
"noWorkflow": "Geen werkstroom",
|
||||||
"removeLinearView": "Verwijder uit lineaire weergave",
|
|
||||||
"workflowTags": "Labels",
|
"workflowTags": "Labels",
|
||||||
"fullyContainNodesHelp": "Knooppunten moeten zich volledig binnen het keuzevak bevinden om te worden gekozen",
|
"fullyContainNodesHelp": "Knooppunten moeten zich volledig binnen het keuzevak bevinden om te worden gekozen",
|
||||||
"workflowValidation": "Validatiefout werkstroom",
|
"workflowValidation": "Validatiefout werkstroom",
|
||||||
@@ -397,14 +377,11 @@
|
|||||||
"unknownField": "Onbekend veld",
|
"unknownField": "Onbekend veld",
|
||||||
"colorCodeEdges": "Kleurgecodeerde randen",
|
"colorCodeEdges": "Kleurgecodeerde randen",
|
||||||
"unknownNode": "Onbekend knooppunt",
|
"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)",
|
"addNodeToolTip": "Voeg knooppunt toe (Shift+A, spatie)",
|
||||||
"loadingNodes": "Bezig met laden van knooppunten...",
|
"loadingNodes": "Bezig met laden van knooppunten...",
|
||||||
"snapToGridHelp": "Lijn knooppunten uit op raster bij verplaatsing",
|
"snapToGridHelp": "Lijn knooppunten uit op raster bij verplaatsing",
|
||||||
"workflowSettings": "Instellingen werkstroomeditor",
|
"workflowSettings": "Instellingen werkstroomeditor",
|
||||||
"addLinearView": "Voeg toe aan lineaire weergave",
|
|
||||||
"nodePack": "Knooppuntpakket",
|
"nodePack": "Knooppuntpakket",
|
||||||
"unknownInput": "Onbekende invoer: {{name}}",
|
|
||||||
"sourceNodeFieldDoesNotExist": "Ongeldige rand: bron-/uitvoerveld {{node}}.{{field}} bestaat niet",
|
"sourceNodeFieldDoesNotExist": "Ongeldige rand: bron-/uitvoerveld {{node}}.{{field}} bestaat niet",
|
||||||
"collectionFieldType": "Verzameling {{name}}",
|
"collectionFieldType": "Verzameling {{name}}",
|
||||||
"deletedInvalidEdge": "Ongeldige hoek {{source}} -> {{target}} verwijderd",
|
"deletedInvalidEdge": "Ongeldige hoek {{source}} -> {{target}} verwijderd",
|
||||||
@@ -419,7 +396,6 @@
|
|||||||
"sourceNodeDoesNotExist": "Ongeldige rand: bron-/uitvoerknooppunt {{node}} bestaat niet",
|
"sourceNodeDoesNotExist": "Ongeldige rand: bron-/uitvoerknooppunt {{node}} bestaat niet",
|
||||||
"unsupportedArrayItemType": "niet-ondersteunde soort van het array-onderdeel \"{{type}}\"",
|
"unsupportedArrayItemType": "niet-ondersteunde soort van het array-onderdeel \"{{type}}\"",
|
||||||
"targetNodeFieldDoesNotExist": "Ongeldige rand: doel-/invoerveld {{node}}.{{field}} bestaat niet",
|
"targetNodeFieldDoesNotExist": "Ongeldige rand: doel-/invoerveld {{node}}.{{field}} bestaat niet",
|
||||||
"reorderLinearView": "Herorden lineaire weergave",
|
|
||||||
"newWorkflowDesc": "Een nieuwe werkstroom aanmaken?",
|
"newWorkflowDesc": "Een nieuwe werkstroom aanmaken?",
|
||||||
"collectionOrScalarFieldType": "Verzameling|scalair {{name}}",
|
"collectionOrScalarFieldType": "Verzameling|scalair {{name}}",
|
||||||
"newWorkflow": "Nieuwe werkstroom",
|
"newWorkflow": "Nieuwe werkstroom",
|
||||||
@@ -734,27 +710,21 @@
|
|||||||
"refinerStart": "Startwaarde verfijning",
|
"refinerStart": "Startwaarde verfijning",
|
||||||
"scheduler": "Planner",
|
"scheduler": "Planner",
|
||||||
"cfgScale": "CFG-schaal",
|
"cfgScale": "CFG-schaal",
|
||||||
"negStylePrompt": "Negatieve-stijlprompt",
|
|
||||||
"noModelsAvailable": "Geen modellen beschikbaar",
|
"noModelsAvailable": "Geen modellen beschikbaar",
|
||||||
"refiner": "Verfijning",
|
"refiner": "Verfijning",
|
||||||
"negAestheticScore": "Negatieve esthetische score",
|
"negAestheticScore": "Negatieve esthetische score",
|
||||||
"denoisingStrength": "Sterkte ontruising",
|
"denoisingStrength": "Sterkte ontruising",
|
||||||
"refinermodel": "Verfijningsmodel",
|
"refinermodel": "Verfijningsmodel",
|
||||||
"posAestheticScore": "Positieve esthetische score",
|
"posAestheticScore": "Positieve esthetische score",
|
||||||
"concatPromptStyle": "Koppelen van prompt en stijl",
|
|
||||||
"loading": "Bezig met laden...",
|
"loading": "Bezig met laden...",
|
||||||
"steps": "Stappen",
|
"steps": "Stappen",
|
||||||
"posStylePrompt": "Positieve-stijlprompt",
|
|
||||||
"freePromptStyle": "Handmatige stijlprompt",
|
|
||||||
"refinerSteps": "Aantal stappen verfijner"
|
"refinerSteps": "Aantal stappen verfijner"
|
||||||
},
|
},
|
||||||
"models": {
|
"models": {
|
||||||
"noMatchingModels": "Geen overeenkomend modellen",
|
"noMatchingModels": "Geen overeenkomend modellen",
|
||||||
"loading": "bezig met laden",
|
"loading": "bezig met laden",
|
||||||
"noMatchingLoRAs": "Geen overeenkomende LoRA's",
|
|
||||||
"noModelsAvailable": "Geen modellen beschikbaar",
|
"noModelsAvailable": "Geen modellen beschikbaar",
|
||||||
"selectModel": "Kies een model",
|
"selectModel": "Kies een model",
|
||||||
"noLoRAsInstalled": "Geen LoRA's geïnstalleerd",
|
|
||||||
"noRefinerModelsInstalled": "Geen SDXL-verfijningsmodellen geïnstalleerd",
|
"noRefinerModelsInstalled": "Geen SDXL-verfijningsmodellen geïnstalleerd",
|
||||||
"defaultVAE": "Standaard-VAE",
|
"defaultVAE": "Standaard-VAE",
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
@@ -822,14 +792,12 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"upscaleMethod": "Opschaalmethode",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"strength": "Sterkte oplossing voor hoge resolutie",
|
"strength": "Sterkte oplossing voor hoge resolutie",
|
||||||
"method": "Methode oplossing voor hoge resolutie",
|
"method": "Methode oplossing voor hoge resolutie",
|
||||||
"enabled": "Oplossing voor hoge resolutie ingeschakeld"
|
"enabled": "Oplossing voor hoge resolutie ingeschakeld"
|
||||||
},
|
},
|
||||||
"hrf": "Oplossing voor hoge resolutie",
|
"hrf": "Oplossing voor hoge resolutie"
|
||||||
"enableHrf": "Schakel oplossing in voor hoge resolutie"
|
|
||||||
},
|
},
|
||||||
"prompt": {
|
"prompt": {
|
||||||
"addPromptTrigger": "Voeg prompttrigger toe",
|
"addPromptTrigger": "Voeg prompttrigger toe",
|
||||||
|
|||||||
@@ -41,11 +41,9 @@
|
|||||||
"somethingWentWrong": "Coś poszło nie tak",
|
"somethingWentWrong": "Coś poszło nie tak",
|
||||||
"green": "Zielony",
|
"green": "Zielony",
|
||||||
"red": "Czerwony",
|
"red": "Czerwony",
|
||||||
"imageFailedToLoad": "Nie można załadować obrazu",
|
|
||||||
"saveAs": "Zapisz jako",
|
"saveAs": "Zapisz jako",
|
||||||
"outputs": "Wyjścia",
|
"outputs": "Wyjścia",
|
||||||
"data": "Dane",
|
"data": "Dane",
|
||||||
"localSystem": "System Lokalny",
|
|
||||||
"t2iAdapter": "Adapter T2I",
|
"t2iAdapter": "Adapter T2I",
|
||||||
"selected": "Zaznaczone",
|
"selected": "Zaznaczone",
|
||||||
"warnings": "Ostrzeżenia",
|
"warnings": "Ostrzeżenia",
|
||||||
@@ -64,12 +62,10 @@
|
|||||||
"openInViewer": "Otwórz podgląd",
|
"openInViewer": "Otwórz podgląd",
|
||||||
"safetensors": "Bezpieczniki",
|
"safetensors": "Bezpieczniki",
|
||||||
"ok": "Ok",
|
"ok": "Ok",
|
||||||
"goTo": "Idź do",
|
|
||||||
"loadingImage": "wczytywanie zdjęcia",
|
"loadingImage": "wczytywanie zdjęcia",
|
||||||
"input": "Wejście",
|
"input": "Wejście",
|
||||||
"view": "Podgląd",
|
"view": "Podgląd",
|
||||||
"learnMore": "Dowiedz się więcej",
|
"learnMore": "Dowiedz się więcej",
|
||||||
"notInstalled": "Nie $t(common.installed)",
|
|
||||||
"loadingModel": "Wczytywanie modelu",
|
"loadingModel": "Wczytywanie modelu",
|
||||||
"postprocessing": "Przetwarzanie końcowe",
|
"postprocessing": "Przetwarzanie końcowe",
|
||||||
"random": "Losowo",
|
"random": "Losowo",
|
||||||
@@ -83,10 +79,8 @@
|
|||||||
"delete": "Usuń",
|
"delete": "Usuń",
|
||||||
"template": "Szablon",
|
"template": "Szablon",
|
||||||
"txt2img": "Tekst na obraz",
|
"txt2img": "Tekst na obraz",
|
||||||
"prevPage": "Poprzednia strona",
|
|
||||||
"file": "Plik",
|
"file": "Plik",
|
||||||
"toResolve": "Do rozwiązania",
|
"toResolve": "Do rozwiązania",
|
||||||
"nextPage": "Następna strona",
|
|
||||||
"unknownError": "Nieznany błąd",
|
"unknownError": "Nieznany błąd",
|
||||||
"placeholderSelectAModel": "Wybierz model",
|
"placeholderSelectAModel": "Wybierz model",
|
||||||
"new": "Nowy",
|
"new": "Nowy",
|
||||||
@@ -99,7 +93,6 @@
|
|||||||
"galleryImageSize": "Rozmiar obrazów",
|
"galleryImageSize": "Rozmiar obrazów",
|
||||||
"gallerySettings": "Ustawienia galerii",
|
"gallerySettings": "Ustawienia galerii",
|
||||||
"autoSwitchNewImages": "Przełączaj na nowe obrazy",
|
"autoSwitchNewImages": "Przełączaj na nowe obrazy",
|
||||||
"noImagesInGallery": "Brak obrazów w galerii",
|
|
||||||
"gallery": "Galeria",
|
"gallery": "Galeria",
|
||||||
"alwaysShowImageSizeBadge": "Zawsze pokazuj odznakę wielkości obrazu",
|
"alwaysShowImageSizeBadge": "Zawsze pokazuj odznakę wielkości obrazu",
|
||||||
"assetsTab": "Pliki, które wrzuciłeś do użytku w twoich projektach.",
|
"assetsTab": "Pliki, które wrzuciłeś do użytku w twoich projektach.",
|
||||||
@@ -128,12 +121,10 @@
|
|||||||
"scaledHeight": "Sk. do wys.",
|
"scaledHeight": "Sk. do wys.",
|
||||||
"infillMethod": "Metoda wypełniania",
|
"infillMethod": "Metoda wypełniania",
|
||||||
"tileSize": "Rozmiar kafelka",
|
"tileSize": "Rozmiar kafelka",
|
||||||
"downloadImage": "Pobierz obraz",
|
|
||||||
"usePrompt": "Skopiuj sugestie",
|
"usePrompt": "Skopiuj sugestie",
|
||||||
"useSeed": "Skopiuj inicjator",
|
"useSeed": "Skopiuj inicjator",
|
||||||
"useAll": "Skopiuj wszystko",
|
"useAll": "Skopiuj wszystko",
|
||||||
"info": "Informacje",
|
"info": "Informacje"
|
||||||
"showOptionsPanel": "Pokaż panel ustawień"
|
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"models": "Modele",
|
"models": "Modele",
|
||||||
@@ -186,8 +177,6 @@
|
|||||||
"selectedForAutoAdd": "Wybrany do automatycznego dodania",
|
"selectedForAutoAdd": "Wybrany do automatycznego dodania",
|
||||||
"deleteBoard": "Usuń tablicę",
|
"deleteBoard": "Usuń tablicę",
|
||||||
"clearSearch": "Usuń historię",
|
"clearSearch": "Usuń historię",
|
||||||
"hideBoards": "Ukryj tablice",
|
|
||||||
"viewBoards": "Zobacz tablice",
|
|
||||||
"addSharedBoard": "Dodaj udostępnioną tablicę",
|
"addSharedBoard": "Dodaj udostępnioną tablicę",
|
||||||
"boards": "Tablice",
|
"boards": "Tablice",
|
||||||
"addPrivateBoard": "Dodaj prywatną tablicę",
|
"addPrivateBoard": "Dodaj prywatną tablicę",
|
||||||
@@ -233,8 +222,7 @@
|
|||||||
"strength": "Moc poprawki wysokiej rozdzielczości",
|
"strength": "Moc poprawki wysokiej rozdzielczości",
|
||||||
"method": "Metoda High Resolution Fix"
|
"method": "Metoda High Resolution Fix"
|
||||||
},
|
},
|
||||||
"hrf": "Poprawka \"Wysoka rozdzielczość\"",
|
"hrf": "Poprawka \"Wysoka rozdzielczość\""
|
||||||
"enableHrf": "Włącz poprawkę wysokiej rozdzielczości"
|
|
||||||
},
|
},
|
||||||
"queue": {
|
"queue": {
|
||||||
"cancelTooltip": "Anuluj aktualną pozycję",
|
"cancelTooltip": "Anuluj aktualną pozycję",
|
||||||
@@ -296,7 +284,6 @@
|
|||||||
"completed": "Zakończono",
|
"completed": "Zakończono",
|
||||||
"item": "Pozycja",
|
"item": "Pozycja",
|
||||||
"failed": "Niepowodzenie",
|
"failed": "Niepowodzenie",
|
||||||
"batchFieldValues": "Masowe Wartości pól",
|
|
||||||
"graphFailedToQueue": "NIe udało się dodać tabeli do kolejki",
|
"graphFailedToQueue": "NIe udało się dodać tabeli do kolejki",
|
||||||
"workflows": "Przepływy pracy",
|
"workflows": "Przepływy pracy",
|
||||||
"next": "Następny",
|
"next": "Następny",
|
||||||
|
|||||||
@@ -17,8 +17,7 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "Tamanho da Imagem",
|
"galleryImageSize": "Tamanho da Imagem",
|
||||||
"gallerySettings": "Configurações de Galeria",
|
"gallerySettings": "Configurações de Galeria",
|
||||||
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente",
|
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente"
|
||||||
"noImagesInGallery": "Sem Imagens na Galeria"
|
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
"modelManager": "Gerente de Modelo",
|
"modelManager": "Gerente de Modelo",
|
||||||
@@ -74,12 +73,10 @@
|
|||||||
"scaledHeight": "A Escalada",
|
"scaledHeight": "A Escalada",
|
||||||
"infillMethod": "Método de Preenchimento",
|
"infillMethod": "Método de Preenchimento",
|
||||||
"tileSize": "Tamanho do Ladrilho",
|
"tileSize": "Tamanho do Ladrilho",
|
||||||
"downloadImage": "Baixar Imagem",
|
|
||||||
"usePrompt": "Usar Prompt",
|
"usePrompt": "Usar Prompt",
|
||||||
"useSeed": "Usar Seed",
|
"useSeed": "Usar Seed",
|
||||||
"useAll": "Usar Todos",
|
"useAll": "Usar Todos",
|
||||||
"info": "Informações",
|
"info": "Informações",
|
||||||
"showOptionsPanel": "Mostrar Painel de Opções",
|
|
||||||
"symmetry": "Simetria",
|
"symmetry": "Simetria",
|
||||||
"copyImage": "Copiar imagem",
|
"copyImage": "Copiar imagem",
|
||||||
"denoisingStrength": "A força de remoção de ruído",
|
"denoisingStrength": "A força de remoção de ruído",
|
||||||
|
|||||||
@@ -17,7 +17,6 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"gallerySettings": "Configurações de Galeria",
|
"gallerySettings": "Configurações de Galeria",
|
||||||
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente",
|
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente",
|
||||||
"noImagesInGallery": "Sem Imagens na Galeria",
|
|
||||||
"galleryImageSize": "Tamanho da Imagem"
|
"galleryImageSize": "Tamanho da Imagem"
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
@@ -69,7 +68,6 @@
|
|||||||
"tileSize": "Tamanho do Ladrilho",
|
"tileSize": "Tamanho do Ladrilho",
|
||||||
"symmetry": "Simetria",
|
"symmetry": "Simetria",
|
||||||
"usePrompt": "Usar Prompt",
|
"usePrompt": "Usar Prompt",
|
||||||
"showOptionsPanel": "Mostrar Painel de Opções",
|
|
||||||
"strength": "Força",
|
"strength": "Força",
|
||||||
"upscaling": "Redimensionando",
|
"upscaling": "Redimensionando",
|
||||||
"scaleBeforeProcessing": "Escala Antes do Processamento",
|
"scaleBeforeProcessing": "Escala Antes do Processamento",
|
||||||
@@ -81,7 +79,6 @@
|
|||||||
"scaledHeight": "A Escalada",
|
"scaledHeight": "A Escalada",
|
||||||
"infillMethod": "Método de Preenchimento",
|
"infillMethod": "Método de Preenchimento",
|
||||||
"copyImage": "Copiar imagem",
|
"copyImage": "Copiar imagem",
|
||||||
"downloadImage": "Descarregar Imagem",
|
|
||||||
"useSeed": "Usar Seed",
|
"useSeed": "Usar Seed",
|
||||||
"useAll": "Usar Todos",
|
"useAll": "Usar Todos",
|
||||||
"info": "Informações"
|
"info": "Informações"
|
||||||
|
|||||||
@@ -38,7 +38,6 @@
|
|||||||
"save": "Сохранить",
|
"save": "Сохранить",
|
||||||
"created": "Создано",
|
"created": "Создано",
|
||||||
"error": "Ошибка",
|
"error": "Ошибка",
|
||||||
"prevPage": "Предыдущая страница",
|
|
||||||
"simple": "Простой",
|
"simple": "Простой",
|
||||||
"ipAdapter": "IP Adapter",
|
"ipAdapter": "IP Adapter",
|
||||||
"installed": "Установлено",
|
"installed": "Установлено",
|
||||||
@@ -49,7 +48,6 @@
|
|||||||
"template": "Шаблон",
|
"template": "Шаблон",
|
||||||
"outputs": "результаты",
|
"outputs": "результаты",
|
||||||
"unknownError": "Неизвестная ошибка",
|
"unknownError": "Неизвестная ошибка",
|
||||||
"imageFailedToLoad": "Невозможно загрузить изображение",
|
|
||||||
"direction": "Направление",
|
"direction": "Направление",
|
||||||
"data": "Данные",
|
"data": "Данные",
|
||||||
"somethingWentWrong": "Что-то пошло не так",
|
"somethingWentWrong": "Что-то пошло не так",
|
||||||
@@ -58,11 +56,9 @@
|
|||||||
"orderBy": "Сортировать по",
|
"orderBy": "Сортировать по",
|
||||||
"copyError": "Ошибка $t(gallery.copy)",
|
"copyError": "Ошибка $t(gallery.copy)",
|
||||||
"learnMore": "Узнать больше",
|
"learnMore": "Узнать больше",
|
||||||
"nextPage": "Следущая страница",
|
|
||||||
"saveAs": "Сохранить как",
|
"saveAs": "Сохранить как",
|
||||||
"input": "Вход",
|
"input": "Вход",
|
||||||
"details": "Детали",
|
"details": "Детали",
|
||||||
"notInstalled": "Нет $t(common.installed)",
|
|
||||||
"or": "или",
|
"or": "или",
|
||||||
"aboutHeading": "Владей своей творческой силой",
|
"aboutHeading": "Владей своей творческой силой",
|
||||||
"red": "Красный",
|
"red": "Красный",
|
||||||
@@ -71,7 +67,6 @@
|
|||||||
"alpha": "Альфа",
|
"alpha": "Альфа",
|
||||||
"toResolve": "Чтоб решить",
|
"toResolve": "Чтоб решить",
|
||||||
"copy": "Копировать",
|
"copy": "Копировать",
|
||||||
"localSystem": "Локальная система",
|
|
||||||
"aboutDesc": "Используя Invoke для работы? Проверьте это:",
|
"aboutDesc": "Используя Invoke для работы? Проверьте это:",
|
||||||
"add": "Добавить",
|
"add": "Добавить",
|
||||||
"beta": "Бета",
|
"beta": "Бета",
|
||||||
@@ -79,7 +74,6 @@
|
|||||||
"positivePrompt": "Позитивный запрос",
|
"positivePrompt": "Позитивный запрос",
|
||||||
"negativePrompt": "Негативный запрос",
|
"negativePrompt": "Негативный запрос",
|
||||||
"editor": "Редактор",
|
"editor": "Редактор",
|
||||||
"goTo": "Перейти к",
|
|
||||||
"tab": "Вкладка",
|
"tab": "Вкладка",
|
||||||
"enabled": "Включено",
|
"enabled": "Включено",
|
||||||
"disabled": "Отключено",
|
"disabled": "Отключено",
|
||||||
@@ -101,7 +95,6 @@
|
|||||||
"galleryImageSize": "Размер изображений",
|
"galleryImageSize": "Размер изображений",
|
||||||
"gallerySettings": "Настройка галереи",
|
"gallerySettings": "Настройка галереи",
|
||||||
"autoSwitchNewImages": "Автоматически выбирать новые",
|
"autoSwitchNewImages": "Автоматически выбирать новые",
|
||||||
"noImagesInGallery": "Изображений нет",
|
|
||||||
"deleteImagePermanent": "Удаленные изображения невозможно восстановить.",
|
"deleteImagePermanent": "Удаленные изображения невозможно восстановить.",
|
||||||
"deleteImage_one": "Удалить изображение",
|
"deleteImage_one": "Удалить изображение",
|
||||||
"deleteImage_few": "Удалить {{count}} изображения",
|
"deleteImage_few": "Удалить {{count}} изображения",
|
||||||
@@ -110,7 +103,6 @@
|
|||||||
"deleteSelection": "Удалить выделенное",
|
"deleteSelection": "Удалить выделенное",
|
||||||
"featuresWillReset": "Если вы удалите это изображение, эти функции будут немедленно сброшены.",
|
"featuresWillReset": "Если вы удалите это изображение, эти функции будут немедленно сброшены.",
|
||||||
"loading": "Загрузка",
|
"loading": "Загрузка",
|
||||||
"unableToLoad": "Невозможно загрузить галерею",
|
|
||||||
"image": "изображение",
|
"image": "изображение",
|
||||||
"drop": "перебросить",
|
"drop": "перебросить",
|
||||||
"downloadSelection": "Скачать выделенное",
|
"downloadSelection": "Скачать выделенное",
|
||||||
@@ -136,7 +128,6 @@
|
|||||||
"compareHelp4": "Нажмите <Kbd>Z</Kbd> или <Kbd>Esc</Kbd> для выхода.",
|
"compareHelp4": "Нажмите <Kbd>Z</Kbd> или <Kbd>Esc</Kbd> для выхода.",
|
||||||
"compareImage": "Сравнить изображение",
|
"compareImage": "Сравнить изображение",
|
||||||
"viewerImage": "Изображение просмотрщика",
|
"viewerImage": "Изображение просмотрщика",
|
||||||
"selectAnImageToCompare": "Выберите изображение для сравнения",
|
|
||||||
"slider": "Слайдер",
|
"slider": "Слайдер",
|
||||||
"sideBySide": "Бок о бок",
|
"sideBySide": "Бок о бок",
|
||||||
"compareHelp1": "Удерживайте <Kbd>Alt</Kbd> при нажатии на изображение в галерее или при помощи клавиш со стрелками, чтобы изменить сравниваемое изображение.",
|
"compareHelp1": "Удерживайте <Kbd>Alt</Kbd> при нажатии на изображение в галерее или при помощи клавиш со стрелками, чтобы изменить сравниваемое изображение.",
|
||||||
@@ -154,11 +145,8 @@
|
|||||||
"exitBoardSearch": "Выйти из поиска досок",
|
"exitBoardSearch": "Выйти из поиска досок",
|
||||||
"go": "Перейти",
|
"go": "Перейти",
|
||||||
"exitSearch": "Выйти из поиска изображений",
|
"exitSearch": "Выйти из поиска изображений",
|
||||||
"jump": "Пыгнуть",
|
|
||||||
"move": "Двигать",
|
"move": "Двигать",
|
||||||
"gallery": "Галерея",
|
"gallery": "Галерея",
|
||||||
"openViewer": "Открыть просмотрщик",
|
|
||||||
"closeViewer": "Закрыть просмотрщик",
|
|
||||||
"imagesTab": "Изображения, созданные и сохраненные в Invoke.",
|
"imagesTab": "Изображения, созданные и сохраненные в Invoke.",
|
||||||
"assetsTab": "Файлы, которые вы загрузили для использования в своих проектах.",
|
"assetsTab": "Файлы, которые вы загрузили для использования в своих проектах.",
|
||||||
"boardsSettings": "Настройки доски",
|
"boardsSettings": "Настройки доски",
|
||||||
@@ -285,10 +273,6 @@
|
|||||||
"title": "Next Layer",
|
"title": "Next Layer",
|
||||||
"desc": "Select the next layer in the list."
|
"desc": "Select the next layer in the list."
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "Set Color to White",
|
|
||||||
"desc": "Set the current tool color to white."
|
|
||||||
},
|
|
||||||
"applyFilter": {
|
"applyFilter": {
|
||||||
"title": "Apply Filter",
|
"title": "Apply Filter",
|
||||||
"desc": "Apply the pending filter to the selected layer."
|
"desc": "Apply the pending filter to the selected layer."
|
||||||
@@ -578,8 +562,6 @@
|
|||||||
"noModelsInstalled": "Нет установленных моделей",
|
"noModelsInstalled": "Нет установленных моделей",
|
||||||
"noModelsInstalledDesc1": "Установите модели с помощью",
|
"noModelsInstalledDesc1": "Установите модели с помощью",
|
||||||
"noMatchingModels": "Нет подходящих моделей",
|
"noMatchingModels": "Нет подходящих моделей",
|
||||||
"ipAdapters": "IP адаптеры",
|
|
||||||
"starterModelsInModelManager": "Стартовые модели можно найти в Менеджере моделей",
|
|
||||||
"learnMoreAboutSupportedModels": "Подробнее о поддерживаемых моделях",
|
"learnMoreAboutSupportedModels": "Подробнее о поддерживаемых моделях",
|
||||||
"t5Encoder": "T5 энкодер",
|
"t5Encoder": "T5 энкодер",
|
||||||
"spandrelImageToImage": "Image to Image (Spandrel)",
|
"spandrelImageToImage": "Image to Image (Spandrel)",
|
||||||
@@ -616,12 +598,10 @@
|
|||||||
"scaledHeight": "Масштаб В",
|
"scaledHeight": "Масштаб В",
|
||||||
"infillMethod": "Способ заполнения",
|
"infillMethod": "Способ заполнения",
|
||||||
"tileSize": "Размер области",
|
"tileSize": "Размер области",
|
||||||
"downloadImage": "Скачать",
|
|
||||||
"usePrompt": "Использовать запрос",
|
"usePrompt": "Использовать запрос",
|
||||||
"useSeed": "Использовать сид",
|
"useSeed": "Использовать сид",
|
||||||
"useAll": "Использовать все",
|
"useAll": "Использовать все",
|
||||||
"info": "Метаданные",
|
"info": "Метаданные",
|
||||||
"showOptionsPanel": "Показать панель настроек",
|
|
||||||
"cancel": {
|
"cancel": {
|
||||||
"cancel": "Отмена"
|
"cancel": "Отмена"
|
||||||
},
|
},
|
||||||
@@ -647,10 +627,6 @@
|
|||||||
"missingFieldTemplate": "Отсутствует шаблон поля",
|
"missingFieldTemplate": "Отсутствует шаблон поля",
|
||||||
"addingImagesTo": "Добавление изображений в",
|
"addingImagesTo": "Добавление изображений в",
|
||||||
"invoke": "Создать",
|
"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",
|
"noFLUXVAEModelSelected": "Для генерации FLUX не выбрана модель VAE",
|
||||||
"noT5EncoderModelSelected": "Для генерации FLUX не выбрана модель T5 энкодера",
|
"noT5EncoderModelSelected": "Для генерации FLUX не выбрана модель T5 энкодера",
|
||||||
"canvasIsFiltering": "Холст фильтруется",
|
"canvasIsFiltering": "Холст фильтруется",
|
||||||
@@ -736,9 +712,6 @@
|
|||||||
"baseModelChangedCleared_few": "Очищено или отключено {{count}} несовместимых подмодели",
|
"baseModelChangedCleared_few": "Очищено или отключено {{count}} несовместимых подмодели",
|
||||||
"baseModelChangedCleared_many": "Очищено или отключено {{count}} несовместимых подмоделей",
|
"baseModelChangedCleared_many": "Очищено или отключено {{count}} несовместимых подмоделей",
|
||||||
"loadedWithWarnings": "Рабочий процесс загружен с предупреждениями",
|
"loadedWithWarnings": "Рабочий процесс загружен с предупреждениями",
|
||||||
"setControlImage": "Установить как контрольное изображение",
|
|
||||||
"setNodeField": "Установить как поле узла",
|
|
||||||
"invalidUpload": "Неверная загрузка",
|
|
||||||
"imageUploaded": "Изображение загружено",
|
"imageUploaded": "Изображение загружено",
|
||||||
"addedToBoard": "Добавлено в активы доски {{name}}",
|
"addedToBoard": "Добавлено в активы доски {{name}}",
|
||||||
"workflowLoaded": "Рабочий процесс загружен",
|
"workflowLoaded": "Рабочий процесс загружен",
|
||||||
@@ -767,21 +740,14 @@
|
|||||||
"sentToCanvas": "Отправить на холст",
|
"sentToCanvas": "Отправить на холст",
|
||||||
"unableToLoadImage": "Невозможно загрузить изображение",
|
"unableToLoadImage": "Невозможно загрузить изображение",
|
||||||
"unableToLoadImageMetadata": "Невозможно загрузить метаданные изображения",
|
"unableToLoadImageMetadata": "Невозможно загрузить метаданные изображения",
|
||||||
"imageSaved": "Изображение сохранено",
|
|
||||||
"stylePresetLoaded": "Предустановка стиля загружена",
|
"stylePresetLoaded": "Предустановка стиля загружена",
|
||||||
"imageNotLoadedDesc": "Не удалось найти изображение",
|
|
||||||
"imageSavingFailed": "Не удалось сохранить изображение",
|
|
||||||
"problemCopyingLayer": "Не удалось скопировать слой",
|
"problemCopyingLayer": "Не удалось скопировать слой",
|
||||||
"unableToLoadStylePreset": "Невозможно загрузить предустановку стиля",
|
"unableToLoadStylePreset": "Невозможно загрузить предустановку стиля",
|
||||||
"layerCopiedToClipboard": "Слой скопирован в буфер обмена",
|
"layerCopiedToClipboard": "Слой скопирован в буфер обмена",
|
||||||
"sentToUpscale": "Отправить на увеличение",
|
"sentToUpscale": "Отправить на увеличение",
|
||||||
"layerSavedToAssets": "Слой сохранен в активах",
|
|
||||||
"linkCopied": "Ссылка скопирована",
|
"linkCopied": "Ссылка скопирована",
|
||||||
"addedToUncategorized": "Добавлено в активы доски $t(boards.uncategorized)",
|
"addedToUncategorized": "Добавлено в активы доски $t(boards.uncategorized)",
|
||||||
"imagesWillBeAddedTo": "Загруженные изображения будут добавлены в активы доски {{boardName}}.",
|
"imagesWillBeAddedTo": "Загруженные изображения будут добавлены в активы доски {{boardName}}."
|
||||||
"uploadFailedInvalidUploadDesc_withCount_one": "Должно быть не более {{count}} изображения в формате PNG или JPEG.",
|
|
||||||
"uploadFailedInvalidUploadDesc_withCount_few": "Должно быть не более {{count}} изображений в формате PNG или JPEG.",
|
|
||||||
"uploadFailedInvalidUploadDesc_withCount_many": "Должно быть не более {{count}} изображений в формате PNG или JPEG."
|
|
||||||
},
|
},
|
||||||
"accessibility": {
|
"accessibility": {
|
||||||
"uploadImage": "Загрузить изображение",
|
"uploadImage": "Загрузить изображение",
|
||||||
@@ -803,15 +769,12 @@
|
|||||||
"zoomInNodes": "Увеличьте масштаб",
|
"zoomInNodes": "Увеличьте масштаб",
|
||||||
"zoomOutNodes": "Уменьшите масштаб",
|
"zoomOutNodes": "Уменьшите масштаб",
|
||||||
"fitViewportNodes": "Уместить вид",
|
"fitViewportNodes": "Уместить вид",
|
||||||
"showLegendNodes": "Показать тип поля",
|
|
||||||
"hideMinimapnodes": "Скрыть миникарту",
|
"hideMinimapnodes": "Скрыть миникарту",
|
||||||
"hideLegendNodes": "Скрыть тип поля",
|
|
||||||
"showMinimapnodes": "Показать миникарту",
|
"showMinimapnodes": "Показать миникарту",
|
||||||
"loadWorkflow": "Загрузить рабочий процесс",
|
"loadWorkflow": "Загрузить рабочий процесс",
|
||||||
"reloadNodeTemplates": "Перезагрузить шаблоны узлов",
|
"reloadNodeTemplates": "Перезагрузить шаблоны узлов",
|
||||||
"downloadWorkflow": "Скачать JSON рабочего процесса",
|
"downloadWorkflow": "Скачать JSON рабочего процесса",
|
||||||
"addNode": "Добавить узел",
|
"addNode": "Добавить узел",
|
||||||
"addLinearView": "Добавить в линейный вид",
|
|
||||||
"animatedEdges": "Анимированные ребра",
|
"animatedEdges": "Анимированные ребра",
|
||||||
"animatedEdgesHelp": "Анимация выбранных ребер и ребер, соединенных с выбранными узлами",
|
"animatedEdgesHelp": "Анимация выбранных ребер и ребер, соединенных с выбранными узлами",
|
||||||
"boolean": "Логические значения",
|
"boolean": "Логические значения",
|
||||||
@@ -823,7 +786,6 @@
|
|||||||
"workflowDescription": "Краткое описание",
|
"workflowDescription": "Краткое описание",
|
||||||
"inputFieldTypeParseError": "Невозможно разобрать тип поля ввода {{node}}.{{field}} ({{message}})",
|
"inputFieldTypeParseError": "Невозможно разобрать тип поля ввода {{node}}.{{field}} ({{message}})",
|
||||||
"unsupportedAnyOfLength": "слишком много элементов объединения ({{count}})",
|
"unsupportedAnyOfLength": "слишком много элементов объединения ({{count}})",
|
||||||
"versionUnknown": " Версия неизвестна",
|
|
||||||
"unsupportedArrayItemType": "неподдерживаемый тип элемента массива \"{{type}}\"",
|
"unsupportedArrayItemType": "неподдерживаемый тип элемента массива \"{{type}}\"",
|
||||||
"noNodeSelected": "Узел не выбран",
|
"noNodeSelected": "Узел не выбран",
|
||||||
"unableToValidateWorkflow": "Невозможно проверить рабочий процесс",
|
"unableToValidateWorkflow": "Невозможно проверить рабочий процесс",
|
||||||
@@ -841,10 +803,8 @@
|
|||||||
"nodeTemplate": "Шаблон узла",
|
"nodeTemplate": "Шаблон узла",
|
||||||
"nodeOpacity": "Непрозрачность узла",
|
"nodeOpacity": "Непрозрачность узла",
|
||||||
"sourceNodeDoesNotExist": "Недопустимое ребро: исходный/выходной узел {{node}} не существует",
|
"sourceNodeDoesNotExist": "Недопустимое ребро: исходный/выходной узел {{node}} не существует",
|
||||||
"unableToLoadWorkflow": "Невозможно загрузить рабочий процесс",
|
|
||||||
"unableToExtractEnumOptions": "невозможно извлечь параметры перечисления",
|
"unableToExtractEnumOptions": "невозможно извлечь параметры перечисления",
|
||||||
"snapToGrid": "Привязка к сетке",
|
"snapToGrid": "Привязка к сетке",
|
||||||
"noFieldsLinearview": "Нет полей, добавленных в линейный вид",
|
|
||||||
"unableToParseFieldType": "невозможно проанализировать тип поля",
|
"unableToParseFieldType": "невозможно проанализировать тип поля",
|
||||||
"nodeSearch": "Поиск узлов",
|
"nodeSearch": "Поиск узлов",
|
||||||
"updateNode": "Обновить узел",
|
"updateNode": "Обновить узел",
|
||||||
@@ -865,9 +825,7 @@
|
|||||||
"edge": "Край",
|
"edge": "Край",
|
||||||
"sourceNodeFieldDoesNotExist": "Неверный край: поле источника/вывода {{node}}.{{field}} не существует",
|
"sourceNodeFieldDoesNotExist": "Неверный край: поле источника/вывода {{node}}.{{field}} не существует",
|
||||||
"cannotDuplicateConnection": "Невозможно создать дубликаты соединений",
|
"cannotDuplicateConnection": "Невозможно создать дубликаты соединений",
|
||||||
"unknownTemplate": "Неизвестный шаблон",
|
|
||||||
"noWorkflow": "Нет рабочего процесса",
|
"noWorkflow": "Нет рабочего процесса",
|
||||||
"removeLinearView": "Удалить из линейного вида",
|
|
||||||
"workflowTags": "Теги",
|
"workflowTags": "Теги",
|
||||||
"fullyContainNodesHelp": "Чтобы узлы были выбраны, они должны полностью находиться в поле выбора",
|
"fullyContainNodesHelp": "Чтобы узлы были выбраны, они должны полностью находиться в поле выбора",
|
||||||
"unableToGetWorkflowVersion": "Не удалось получить версию схемы рабочего процесса",
|
"unableToGetWorkflowVersion": "Не удалось получить версию схемы рабочего процесса",
|
||||||
@@ -900,7 +858,6 @@
|
|||||||
"colorCodeEdges": "Ребра с цветовой кодировкой",
|
"colorCodeEdges": "Ребра с цветовой кодировкой",
|
||||||
"unknownNode": "Неизвестный узел",
|
"unknownNode": "Неизвестный узел",
|
||||||
"targetNodeDoesNotExist": "Недопустимое ребро: целевой/входной узел {{node}} не существует",
|
"targetNodeDoesNotExist": "Недопустимое ребро: целевой/входной узел {{node}} не существует",
|
||||||
"mismatchedVersion": "Недопустимый узел: узел {{node}} типа {{type}} имеет несоответствующую версию (попробовать обновить?)",
|
|
||||||
"unknownFieldType": "$t(nodes.unknownField) тип: {{type}}",
|
"unknownFieldType": "$t(nodes.unknownField) тип: {{type}}",
|
||||||
"collectionOrScalarFieldType": "{{name}} (Один или коллекция)",
|
"collectionOrScalarFieldType": "{{name}} (Один или коллекция)",
|
||||||
"betaDesc": "Этот вызов находится в бета-версии. Пока он не станет стабильным, в нем могут происходить изменения при обновлении приложений. Мы планируем поддерживать этот вызов в течение длительного времени.",
|
"betaDesc": "Этот вызов находится в бета-версии. Пока он не станет стабильным, в нем могут происходить изменения при обновлении приложений. Мы планируем поддерживать этот вызов в течение длительного времени.",
|
||||||
@@ -909,14 +866,12 @@
|
|||||||
"snapToGridHelp": "Привязка узлов к сетке при перемещении",
|
"snapToGridHelp": "Привязка узлов к сетке при перемещении",
|
||||||
"workflowSettings": "Настройки редактора рабочих процессов",
|
"workflowSettings": "Настройки редактора рабочих процессов",
|
||||||
"deletedInvalidEdge": "Удалено недопустимое ребро {{source}} -> {{target}}",
|
"deletedInvalidEdge": "Удалено недопустимое ребро {{source}} -> {{target}}",
|
||||||
"unknownInput": "Неизвестный вход: {{name}}",
|
|
||||||
"newWorkflow": "Новый рабочий процесс",
|
"newWorkflow": "Новый рабочий процесс",
|
||||||
"newWorkflowDesc": "Создать новый рабочий процесс?",
|
"newWorkflowDesc": "Создать новый рабочий процесс?",
|
||||||
"clearWorkflow": "Очистить рабочий процесс",
|
"clearWorkflow": "Очистить рабочий процесс",
|
||||||
"newWorkflowDesc2": "Текущий рабочий процесс имеет несохраненные изменения.",
|
"newWorkflowDesc2": "Текущий рабочий процесс имеет несохраненные изменения.",
|
||||||
"clearWorkflowDesc": "Очистить этот рабочий процесс и создать новый?",
|
"clearWorkflowDesc": "Очистить этот рабочий процесс и создать новый?",
|
||||||
"clearWorkflowDesc2": "Текущий рабочий процесс имеет несохраненные измерения.",
|
"clearWorkflowDesc2": "Текущий рабочий процесс имеет несохраненные измерения.",
|
||||||
"reorderLinearView": "Изменить порядок линейного просмотра",
|
|
||||||
"viewMode": "Использовать в линейном представлении",
|
"viewMode": "Использовать в линейном представлении",
|
||||||
"editMode": "Открыть в редакторе узлов",
|
"editMode": "Открыть в редакторе узлов",
|
||||||
"resetToDefaultValue": "Сбросить к стандартному значкнию",
|
"resetToDefaultValue": "Сбросить к стандартному значкнию",
|
||||||
@@ -978,8 +933,6 @@
|
|||||||
"addPrivateBoard": "Добавить личную доску",
|
"addPrivateBoard": "Добавить личную доску",
|
||||||
"private": "Личные доски",
|
"private": "Личные доски",
|
||||||
"shared": "Общие доски",
|
"shared": "Общие доски",
|
||||||
"hideBoards": "Скрыть доски",
|
|
||||||
"viewBoards": "Просмотреть доски",
|
|
||||||
"noBoards": "Нет досок {{boardType}}",
|
"noBoards": "Нет досок {{boardType}}",
|
||||||
"deletedPrivateBoardsCannotbeRestored": "Удаленные доски не могут быть восстановлены. Выбор «Удалить только доску» переведет изображения в приватное состояние без категории для создателя изображения.",
|
"deletedPrivateBoardsCannotbeRestored": "Удаленные доски не могут быть восстановлены. Выбор «Удалить только доску» переведет изображения в приватное состояние без категории для создателя изображения.",
|
||||||
"updateBoardError": "Ошибка обновления доски"
|
"updateBoardError": "Ошибка обновления доски"
|
||||||
@@ -1408,8 +1361,6 @@
|
|||||||
"noRecallParameters": "Параметры для вызова не найдены",
|
"noRecallParameters": "Параметры для вызова не найдены",
|
||||||
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
||||||
"parameterSet": "Параметр {{parameter}} установлен",
|
"parameterSet": "Параметр {{parameter}} установлен",
|
||||||
"parsingFailed": "Не удалось выполнить синтаксический анализ",
|
|
||||||
"recallParameter": "Отозвать {{label}}",
|
|
||||||
"allPrompts": "Все запросы",
|
"allPrompts": "Все запросы",
|
||||||
"imageDimensions": "Размеры изображения",
|
"imageDimensions": "Размеры изображения",
|
||||||
"canvasV2Metadata": "Холст",
|
"canvasV2Metadata": "Холст",
|
||||||
@@ -1460,7 +1411,6 @@
|
|||||||
"next": "Следующий",
|
"next": "Следующий",
|
||||||
"cancelBatch": "Отменить пакет",
|
"cancelBatch": "Отменить пакет",
|
||||||
"back": "задний",
|
"back": "задний",
|
||||||
"batchFieldValues": "Пакетные значения полей",
|
|
||||||
"cancel": "Отмена",
|
"cancel": "Отмена",
|
||||||
"session": "Сессия",
|
"session": "Сессия",
|
||||||
"time": "Время",
|
"time": "Время",
|
||||||
@@ -1495,18 +1445,14 @@
|
|||||||
"refinerStart": "Запуск доработчика",
|
"refinerStart": "Запуск доработчика",
|
||||||
"scheduler": "Планировщик",
|
"scheduler": "Планировщик",
|
||||||
"cfgScale": "Шкала точности (CFG)",
|
"cfgScale": "Шкала точности (CFG)",
|
||||||
"negStylePrompt": "Негативный запрос стиля",
|
|
||||||
"noModelsAvailable": "Нет доступных моделей",
|
"noModelsAvailable": "Нет доступных моделей",
|
||||||
"refiner": "Доработчик",
|
"refiner": "Доработчик",
|
||||||
"negAestheticScore": "Отрицательная эстетическая оценка",
|
"negAestheticScore": "Отрицательная эстетическая оценка",
|
||||||
"denoisingStrength": "Шумоподавление",
|
"denoisingStrength": "Шумоподавление",
|
||||||
"refinermodel": "Дорабатывающая модель",
|
"refinermodel": "Дорабатывающая модель",
|
||||||
"posAestheticScore": "Положительная эстетическая оценка",
|
"posAestheticScore": "Положительная эстетическая оценка",
|
||||||
"concatPromptStyle": "Связывание запроса и стиля",
|
|
||||||
"loading": "Загрузка...",
|
"loading": "Загрузка...",
|
||||||
"steps": "Шаги",
|
"steps": "Шаги",
|
||||||
"posStylePrompt": "Запрос стиля",
|
|
||||||
"freePromptStyle": "Ручной запрос стиля",
|
|
||||||
"refinerSteps": "Шаги доработчика"
|
"refinerSteps": "Шаги доработчика"
|
||||||
},
|
},
|
||||||
"invocationCache": {
|
"invocationCache": {
|
||||||
@@ -1531,20 +1477,15 @@
|
|||||||
"workflowEditorMenu": "Меню редактора рабочего процесса",
|
"workflowEditorMenu": "Меню редактора рабочего процесса",
|
||||||
"workflowName": "Имя рабочего процесса",
|
"workflowName": "Имя рабочего процесса",
|
||||||
"saveWorkflow": "Сохранить рабочий процесс",
|
"saveWorkflow": "Сохранить рабочий процесс",
|
||||||
"openWorkflow": "Открытый рабочий процесс",
|
|
||||||
"clearWorkflowSearchFilter": "Очистить фильтр поиска рабочих процессов",
|
|
||||||
"workflowLibrary": "Библиотека",
|
"workflowLibrary": "Библиотека",
|
||||||
"downloadWorkflow": "Сохранить в файл",
|
"downloadWorkflow": "Сохранить в файл",
|
||||||
"workflowSaved": "Рабочий процесс сохранен",
|
"workflowSaved": "Рабочий процесс сохранен",
|
||||||
"unnamedWorkflow": "Безымянный рабочий процесс",
|
"unnamedWorkflow": "Безымянный рабочий процесс",
|
||||||
"savingWorkflow": "Сохранение рабочего процесса...",
|
"savingWorkflow": "Сохранение рабочего процесса...",
|
||||||
"problemLoading": "Проблема с загрузкой рабочих процессов",
|
|
||||||
"loading": "Загрузка рабочих процессов",
|
"loading": "Загрузка рабочих процессов",
|
||||||
"searchWorkflows": "Поиск рабочих процессов",
|
|
||||||
"problemSavingWorkflow": "Проблема с сохранением рабочего процесса",
|
"problemSavingWorkflow": "Проблема с сохранением рабочего процесса",
|
||||||
"deleteWorkflow": "Удалить рабочий процесс",
|
"deleteWorkflow": "Удалить рабочий процесс",
|
||||||
"workflows": "Рабочие процессы",
|
"workflows": "Рабочие процессы",
|
||||||
"noDescription": "Без описания",
|
|
||||||
"uploadWorkflow": "Загрузить из файла",
|
"uploadWorkflow": "Загрузить из файла",
|
||||||
"newWorkflowCreated": "Создан новый рабочий процесс",
|
"newWorkflowCreated": "Создан новый рабочий процесс",
|
||||||
"saveWorkflowToProject": "Сохранить рабочий процесс в проект",
|
"saveWorkflowToProject": "Сохранить рабочий процесс в проект",
|
||||||
@@ -1560,9 +1501,6 @@
|
|||||||
"convertGraph": "Конвертировать график",
|
"convertGraph": "Конвертировать график",
|
||||||
"loadFromGraph": "Загрузка рабочего процесса из графика",
|
"loadFromGraph": "Загрузка рабочего процесса из графика",
|
||||||
"autoLayout": "Автоматическое расположение",
|
"autoLayout": "Автоматическое расположение",
|
||||||
"userWorkflows": "Пользовательские рабочие процессы",
|
|
||||||
"projectWorkflows": "Рабочие процессы проекта",
|
|
||||||
"defaultWorkflows": "Стандартные рабочие процессы",
|
|
||||||
"deleteWorkflow2": "Вы уверены, что хотите удалить этот рабочий процесс? Это нельзя отменить.",
|
"deleteWorkflow2": "Вы уверены, что хотите удалить этот рабочий процесс? Это нельзя отменить.",
|
||||||
"chooseWorkflowFromLibrary": "Выбрать рабочий процесс из библиотеки",
|
"chooseWorkflowFromLibrary": "Выбрать рабочий процесс из библиотеки",
|
||||||
"edit": "Редактировать",
|
"edit": "Редактировать",
|
||||||
@@ -1572,8 +1510,6 @@
|
|||||||
"delete": "Удалить"
|
"delete": "Удалить"
|
||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"enableHrf": "Включить исправление высокого разрешения",
|
|
||||||
"upscaleMethod": "Метод увеличения",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"strength": "Сила исправления высокого разрешения",
|
"strength": "Сила исправления высокого разрешения",
|
||||||
"enabled": "Исправление высокого разрешения включено",
|
"enabled": "Исправление высокого разрешения включено",
|
||||||
@@ -1584,12 +1520,10 @@
|
|||||||
"models": {
|
"models": {
|
||||||
"noMatchingModels": "Нет подходящих моделей",
|
"noMatchingModels": "Нет подходящих моделей",
|
||||||
"loading": "загрузка",
|
"loading": "загрузка",
|
||||||
"noMatchingLoRAs": "Нет подходящих LoRA",
|
|
||||||
"noModelsAvailable": "Нет доступных моделей",
|
"noModelsAvailable": "Нет доступных моделей",
|
||||||
"addLora": "Добавить LoRA",
|
"addLora": "Добавить LoRA",
|
||||||
"selectModel": "Выберите модель",
|
"selectModel": "Выберите модель",
|
||||||
"noRefinerModelsInstalled": "Дорабатывающие модели SDXL не установлены",
|
"noRefinerModelsInstalled": "Дорабатывающие модели SDXL не установлены",
|
||||||
"noLoRAsInstalled": "Нет установленных LoRA",
|
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
"defaultVAE": "Стандартное VAE",
|
"defaultVAE": "Стандартное VAE",
|
||||||
"concepts": "LoRA"
|
"concepts": "LoRA"
|
||||||
@@ -1624,7 +1558,6 @@
|
|||||||
"moveForward": "Переместить вперёд",
|
"moveForward": "Переместить вперёд",
|
||||||
"moveBackward": "Переместить назад",
|
"moveBackward": "Переместить назад",
|
||||||
"autoNegative": "Авто негатив",
|
"autoNegative": "Авто негатив",
|
||||||
"deletePrompt": "Удалить запрос",
|
|
||||||
"rectangle": "Прямоугольник",
|
"rectangle": "Прямоугольник",
|
||||||
"addNegativePrompt": "Добавить $t(controlLayers.negativePrompt)",
|
"addNegativePrompt": "Добавить $t(controlLayers.negativePrompt)",
|
||||||
"regionalGuidance": "Региональная точность",
|
"regionalGuidance": "Региональная точность",
|
||||||
@@ -1794,7 +1727,6 @@
|
|||||||
},
|
},
|
||||||
"addReferenceImage": "Добавить $t(controlLayers.referenceImage)",
|
"addReferenceImage": "Добавить $t(controlLayers.referenceImage)",
|
||||||
"inpaintMask": "Маска перерисовки",
|
"inpaintMask": "Маска перерисовки",
|
||||||
"sendToGalleryDesc": "При нажатии кнопки Invoke создается изображение и сохраняется в вашей галерее.",
|
|
||||||
"sendToCanvas": "Отправить на холст",
|
"sendToCanvas": "Отправить на холст",
|
||||||
"regionalGuidance_withCount_one": "$t(controlLayers.regionalGuidance)",
|
"regionalGuidance_withCount_one": "$t(controlLayers.regionalGuidance)",
|
||||||
"regionalGuidance_withCount_few": "Региональных точности",
|
"regionalGuidance_withCount_few": "Региональных точности",
|
||||||
@@ -1806,7 +1738,6 @@
|
|||||||
"inpaintMask_withCount_one": "$t(controlLayers.inpaintMask)",
|
"inpaintMask_withCount_one": "$t(controlLayers.inpaintMask)",
|
||||||
"inpaintMask_withCount_few": "Маски перерисовки",
|
"inpaintMask_withCount_few": "Маски перерисовки",
|
||||||
"inpaintMask_withCount_many": "Масок перерисовки",
|
"inpaintMask_withCount_many": "Масок перерисовки",
|
||||||
"globalReferenceImages_withCount_visible": "Глобальные эталонные изображения ({{count}})",
|
|
||||||
"controlMode": {
|
"controlMode": {
|
||||||
"prompt": "Запрос",
|
"prompt": "Запрос",
|
||||||
"controlMode": "Режим контроля",
|
"controlMode": "Режим контроля",
|
||||||
@@ -1842,7 +1773,6 @@
|
|||||||
"pullBboxIntoReferenceImage": "Поместить рамку в эталонное изображение",
|
"pullBboxIntoReferenceImage": "Поместить рамку в эталонное изображение",
|
||||||
"enableAutoNegative": "Включить авто негатив",
|
"enableAutoNegative": "Включить авто негатив",
|
||||||
"maskFill": "Заполнение маски",
|
"maskFill": "Заполнение маски",
|
||||||
"viewProgressInViewer": "Просматривайте прогресс и результаты в <Btn>Просмотрщике изображений</Btn>.",
|
|
||||||
"tool": {
|
"tool": {
|
||||||
"move": "Двигать",
|
"move": "Двигать",
|
||||||
"bbox": "Ограничительная рамка",
|
"bbox": "Ограничительная рамка",
|
||||||
@@ -1853,18 +1783,10 @@
|
|||||||
"colorPicker": "Подборщик цветов"
|
"colorPicker": "Подборщик цветов"
|
||||||
},
|
},
|
||||||
"rasterLayer": "Растровый слой",
|
"rasterLayer": "Растровый слой",
|
||||||
"sendingToCanvas": "Постановка генераций на холст",
|
|
||||||
"rasterLayers_withCount_visible": "Растровые слои ({{count}})",
|
|
||||||
"regionalGuidance_withCount_hidden": "Региональная точность ({{count}} скрыто)",
|
|
||||||
"enableTransparencyEffect": "Включить эффект прозрачности",
|
"enableTransparencyEffect": "Включить эффект прозрачности",
|
||||||
"hidingType": "Скрыть {{type}}",
|
"hidingType": "Скрыть {{type}}",
|
||||||
"addRegionalGuidance": "Добавить $t(controlLayers.regionalGuidance)",
|
"addRegionalGuidance": "Добавить $t(controlLayers.regionalGuidance)",
|
||||||
"sendingToGallery": "Отправка генераций в галерею",
|
|
||||||
"viewProgressOnCanvas": "Просматривайте прогресс и результаты этапов на <Btn>Холсте</Btn>.",
|
|
||||||
"controlLayers_withCount_hidden": "Контрольные слои ({{count}} скрыто)",
|
|
||||||
"rasterLayers_withCount_hidden": "Растровые слои ({{count}} скрыто)",
|
|
||||||
"deleteSelected": "Удалить выбранное",
|
"deleteSelected": "Удалить выбранное",
|
||||||
"stagingOnCanvas": "Постановка изображений на",
|
|
||||||
"pullBboxIntoLayer": "Поместить рамку в слой",
|
"pullBboxIntoLayer": "Поместить рамку в слой",
|
||||||
"locked": "Заблокировано",
|
"locked": "Заблокировано",
|
||||||
"replaceLayer": "Заменить слой",
|
"replaceLayer": "Заменить слой",
|
||||||
@@ -1873,16 +1795,10 @@
|
|||||||
"addRasterLayer": "Добавить $t(controlLayers.rasterLayer)",
|
"addRasterLayer": "Добавить $t(controlLayers.rasterLayer)",
|
||||||
"addControlLayer": "Добавить $t(controlLayers.controlLayer)",
|
"addControlLayer": "Добавить $t(controlLayers.controlLayer)",
|
||||||
"addInpaintMask": "Добавить $t(controlLayers.inpaintMask)",
|
"addInpaintMask": "Добавить $t(controlLayers.inpaintMask)",
|
||||||
"inpaintMasks_withCount_hidden": "Маски перерисовки ({{count}} скрыто)",
|
|
||||||
"regionalGuidance_withCount_visible": "Региональная точность ({{count}})",
|
|
||||||
"newGallerySessionDesc": "Это очистит холст и все настройки, кроме выбранной модели. Генерации будут отправлены в галерею.",
|
|
||||||
"newCanvasSession": "Новая сессия холста",
|
|
||||||
"newCanvasSessionDesc": "Это очистит холст и все настройки, кроме выбора модели. Генерации будут размещены на холсте.",
|
|
||||||
"cropLayerToBbox": "Обрезать слой по ограничительной рамке",
|
"cropLayerToBbox": "Обрезать слой по ограничительной рамке",
|
||||||
"clipToBbox": "Обрезка штрихов в рамке",
|
"clipToBbox": "Обрезка штрихов в рамке",
|
||||||
"outputOnlyMaskedRegions": "Вывод только маскированных областей",
|
"outputOnlyMaskedRegions": "Вывод только маскированных областей",
|
||||||
"duplicate": "Дублировать",
|
"duplicate": "Дублировать",
|
||||||
"inpaintMasks_withCount_visible": "Маски перерисовки ({{count}})",
|
|
||||||
"layer_one": "Слой",
|
"layer_one": "Слой",
|
||||||
"layer_few": "Слоя",
|
"layer_few": "Слоя",
|
||||||
"layer_many": "Слоев",
|
"layer_many": "Слоев",
|
||||||
@@ -1901,33 +1817,20 @@
|
|||||||
},
|
},
|
||||||
"disableAutoNegative": "Отключить авто негатив",
|
"disableAutoNegative": "Отключить авто негатив",
|
||||||
"deleteReferenceImage": "Удалить эталонное изображение",
|
"deleteReferenceImage": "Удалить эталонное изображение",
|
||||||
"controlLayers_withCount_visible": "Контрольные слои ({{count}})",
|
|
||||||
"rasterLayer_withCount_one": "$t(controlLayers.rasterLayer)",
|
"rasterLayer_withCount_one": "$t(controlLayers.rasterLayer)",
|
||||||
"rasterLayer_withCount_few": "Растровых слоя",
|
"rasterLayer_withCount_few": "Растровых слоя",
|
||||||
"rasterLayer_withCount_many": "Растровых слоев",
|
"rasterLayer_withCount_many": "Растровых слоев",
|
||||||
"transparency": "Прозрачность",
|
"transparency": "Прозрачность",
|
||||||
"weight": "Вес",
|
"weight": "Вес",
|
||||||
"newGallerySession": "Новая сессия галереи",
|
|
||||||
"sendToCanvasDesc": "Нажатие кнопки Invoke отображает вашу текущую работу на холсте.",
|
|
||||||
"globalReferenceImages_withCount_hidden": "Глобальные эталонные изображения ({{count}} скрыто)",
|
|
||||||
"layer_withCount_one": "Слой ({{count}})",
|
|
||||||
"layer_withCount_few": "Слои ({{count}})",
|
|
||||||
"layer_withCount_many": "Слои ({{count}})",
|
|
||||||
"disableTransparencyEffect": "Отключить эффект прозрачности",
|
"disableTransparencyEffect": "Отключить эффект прозрачности",
|
||||||
"showingType": "Показать {{type}}",
|
"showingType": "Показать {{type}}",
|
||||||
"dynamicGrid": "Динамическая сетка",
|
"dynamicGrid": "Динамическая сетка",
|
||||||
"logDebugInfo": "Писать отладочную информацию",
|
"logDebugInfo": "Писать отладочную информацию",
|
||||||
"unlocked": "Разблокировано",
|
"unlocked": "Разблокировано",
|
||||||
"showProgressOnCanvas": "Показать прогресс на холсте",
|
"showProgressOnCanvas": "Показать прогресс на холсте",
|
||||||
"globalReferenceImage_withCount_one": "$t(controlLayers.globalReferenceImage)",
|
|
||||||
"globalReferenceImage_withCount_few": "Глобальных эталонных изображения",
|
|
||||||
"globalReferenceImage_withCount_many": "Глобальных эталонных изображений",
|
|
||||||
"regionalReferenceImage": "Региональное эталонное изображение",
|
"regionalReferenceImage": "Региональное эталонное изображение",
|
||||||
"globalReferenceImage": "Глобальное эталонное изображение",
|
"globalReferenceImage": "Глобальное эталонное изображение",
|
||||||
"sendToGallery": "Отправить в галерею",
|
"referenceImage": "Эталонное изображение"
|
||||||
"referenceImage": "Эталонное изображение",
|
|
||||||
"addGlobalReferenceImage": "Добавить $t(controlLayers.globalReferenceImage)",
|
|
||||||
"newImg2ImgCanvasFromImage": "Новое img2img из изображения"
|
|
||||||
},
|
},
|
||||||
"ui": {
|
"ui": {
|
||||||
"tabs": {
|
"tabs": {
|
||||||
|
|||||||
@@ -28,7 +28,6 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "Bildstorlek",
|
"galleryImageSize": "Bildstorlek",
|
||||||
"gallerySettings": "Galleriinställningar",
|
"gallerySettings": "Galleriinställningar",
|
||||||
"noImagesInGallery": "Inga bilder i galleriet",
|
|
||||||
"autoSwitchNewImages": "Ändra automatiskt till nya bilder"
|
"autoSwitchNewImages": "Ändra automatiskt till nya bilder"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,12 +36,10 @@
|
|||||||
"communityLabel": "Topluluk",
|
"communityLabel": "Topluluk",
|
||||||
"back": "Geri",
|
"back": "Geri",
|
||||||
"areYouSure": "Emin misiniz?",
|
"areYouSure": "Emin misiniz?",
|
||||||
"notInstalled": "$t(common.installed) Değil",
|
|
||||||
"openInNewTab": "Yeni Sekmede Aç",
|
"openInNewTab": "Yeni Sekmede Aç",
|
||||||
"aboutHeading": "Yaratıcı Gücünüzün Sahibi Olun",
|
"aboutHeading": "Yaratıcı Gücünüzün Sahibi Olun",
|
||||||
"load": "Yükle",
|
"load": "Yükle",
|
||||||
"loading": "Yükleniyor",
|
"loading": "Yükleniyor",
|
||||||
"localSystem": "Yerel Sistem",
|
|
||||||
"inpaint": "içboyama",
|
"inpaint": "içboyama",
|
||||||
"modelManager": "Model Yöneticisi",
|
"modelManager": "Model Yöneticisi",
|
||||||
"orderBy": "Sırala",
|
"orderBy": "Sırala",
|
||||||
@@ -65,11 +63,8 @@
|
|||||||
"format": "biçim",
|
"format": "biçim",
|
||||||
"details": "Ayrıntılar",
|
"details": "Ayrıntılar",
|
||||||
"error": "Hata",
|
"error": "Hata",
|
||||||
"imageFailedToLoad": "Görsel Yüklenemedi",
|
|
||||||
"safetensors": "Safetensors",
|
"safetensors": "Safetensors",
|
||||||
"upload": "Yükle",
|
"upload": "Yükle",
|
||||||
"nextPage": "Sonraki Sayfa",
|
|
||||||
"prevPage": "Önceki Sayfa",
|
|
||||||
"dontAskMeAgain": "Bir daha sorma",
|
"dontAskMeAgain": "Bir daha sorma",
|
||||||
"delete": "Kaldır",
|
"delete": "Kaldır",
|
||||||
"direction": "Yön",
|
"direction": "Yön",
|
||||||
@@ -181,7 +176,6 @@
|
|||||||
"session": "Oturum",
|
"session": "Oturum",
|
||||||
"batchQueued": "Toplu İş Sıraya Alındı",
|
"batchQueued": "Toplu İş Sıraya Alındı",
|
||||||
"notReady": "Sıraya Alınamadı",
|
"notReady": "Sıraya Alınamadı",
|
||||||
"batchFieldValues": "Toplu İş Değişkenleri",
|
|
||||||
"graphFailedToQueue": "Çizge sıraya alınamadı",
|
"graphFailedToQueue": "Çizge sıraya alınamadı",
|
||||||
"graphQueued": "Çizge sıraya alındı"
|
"graphQueued": "Çizge sıraya alındı"
|
||||||
},
|
},
|
||||||
@@ -207,12 +201,10 @@
|
|||||||
"image": "görsel",
|
"image": "görsel",
|
||||||
"galleryImageSize": "Görsel Boyutu",
|
"galleryImageSize": "Görsel Boyutu",
|
||||||
"copy": "Kopyala",
|
"copy": "Kopyala",
|
||||||
"noImagesInGallery": "Gösterilecek Görsel Yok",
|
|
||||||
"autoSwitchNewImages": "Yeni Görseli Biter Bitmez Gör",
|
"autoSwitchNewImages": "Yeni Görseli Biter Bitmez Gör",
|
||||||
"currentlyInUse": "Bu görsel şurada kullanımda:",
|
"currentlyInUse": "Bu görsel şurada kullanımda:",
|
||||||
"deleteImage_one": "Görseli Sil",
|
"deleteImage_one": "Görseli Sil",
|
||||||
"deleteImage_other": "",
|
"deleteImage_other": "",
|
||||||
"unableToLoad": "Galeri Yüklenemedi",
|
|
||||||
"downloadSelection": "Seçileni İndir",
|
"downloadSelection": "Seçileni İndir",
|
||||||
"dropOrUpload": "$t(gallery.drop) ya da Yükle",
|
"dropOrUpload": "$t(gallery.drop) ya da Yükle",
|
||||||
"dropToUpload": "Yüklemek için $t(gallery.drop)",
|
"dropToUpload": "Yüklemek için $t(gallery.drop)",
|
||||||
@@ -220,13 +212,11 @@
|
|||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"hrf": "Yüksek Çözünürlük Kürü",
|
"hrf": "Yüksek Çözünürlük Kürü",
|
||||||
"enableHrf": "Yüksek Çözünürlük Kürünü Aç",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"enabled": "Yüksek Çözünürlük Kürü Açık",
|
"enabled": "Yüksek Çözünürlük Kürü Açık",
|
||||||
"strength": "Yüksek Çözünürlük Kürü Etkisi",
|
"strength": "Yüksek Çözünürlük Kürü Etkisi",
|
||||||
"method": "Yüksek Çözünürlük Kürü Yöntemi"
|
"method": "Yüksek Çözünürlük Kürü Yöntemi"
|
||||||
},
|
}
|
||||||
"upscaleMethod": "Büyütme Yöntemi"
|
|
||||||
},
|
},
|
||||||
"hotkeys": {
|
"hotkeys": {
|
||||||
"noHotkeysFound": "Kısayol Tuşu Bulanamadı",
|
"noHotkeysFound": "Kısayol Tuşu Bulanamadı",
|
||||||
@@ -256,7 +246,6 @@
|
|||||||
"unknownErrorValidatingWorkflow": "İş akışını doğrulamada bilinmeyen bir sorun",
|
"unknownErrorValidatingWorkflow": "İş akışını doğrulamada bilinmeyen bir sorun",
|
||||||
"unableToGetWorkflowVersion": "İş akışı sürümüne ulaşılamadı",
|
"unableToGetWorkflowVersion": "İş akışı sürümüne ulaşılamadı",
|
||||||
"newWorkflowDesc2": "Geçerli iş akışında kaydedilmemiş değişiklikler var.",
|
"newWorkflowDesc2": "Geçerli iş akışında kaydedilmemiş değişiklikler var.",
|
||||||
"unableToLoadWorkflow": "İş Akışı Yüklenemedi",
|
|
||||||
"cannotConnectInputToInput": "Giriş girişe bağlanamaz",
|
"cannotConnectInputToInput": "Giriş girişe bağlanamaz",
|
||||||
"zoomInNodes": "Yakınlaştır",
|
"zoomInNodes": "Yakınlaştır",
|
||||||
"boolean": "Boole Değeri",
|
"boolean": "Boole Değeri",
|
||||||
@@ -267,16 +256,12 @@
|
|||||||
"cannotDuplicateConnection": "Kopya bağlantılar yaratılamaz"
|
"cannotDuplicateConnection": "Kopya bağlantılar yaratılamaz"
|
||||||
},
|
},
|
||||||
"workflows": {
|
"workflows": {
|
||||||
"searchWorkflows": "İş Akışlarında Ara",
|
|
||||||
"workflowName": "İş Akışı Adı",
|
"workflowName": "İş Akışı Adı",
|
||||||
"problemSavingWorkflow": "İş Akışını Kaydetmede Sorun",
|
"problemSavingWorkflow": "İş Akışını Kaydetmede Sorun",
|
||||||
"saveWorkflow": "İş Akışını Kaydet",
|
"saveWorkflow": "İş Akışını Kaydet",
|
||||||
"uploadWorkflow": "Dosyadan Yükle",
|
"uploadWorkflow": "Dosyadan Yükle",
|
||||||
"newWorkflowCreated": "Yeni İş Akışı Yaratıldı",
|
"newWorkflowCreated": "Yeni İş Akışı Yaratıldı",
|
||||||
"problemLoading": "İş Akışlarını Yüklemede Sorun",
|
|
||||||
"loading": "İş Akışları Yükleniyor",
|
"loading": "İş Akışları Yükleniyor",
|
||||||
"noDescription": "Tanımsız",
|
|
||||||
"clearWorkflowSearchFilter": "İş Akışı Aramasını Resetle",
|
|
||||||
"workflowEditorMenu": "İş Akışı Düzenleyici Menüsü",
|
"workflowEditorMenu": "İş Akışı Düzenleyici Menüsü",
|
||||||
"downloadWorkflow": "İndir",
|
"downloadWorkflow": "İndir",
|
||||||
"saveWorkflowAs": "İş Akışını Farklı Kaydet",
|
"saveWorkflowAs": "İş Akışını Farklı Kaydet",
|
||||||
@@ -328,7 +313,6 @@
|
|||||||
"noiseThreshold": "Gürültü Eşiği",
|
"noiseThreshold": "Gürültü Eşiği",
|
||||||
"seed": "Tohum",
|
"seed": "Tohum",
|
||||||
"imageActions": "Görsel İşlemleri",
|
"imageActions": "Görsel İşlemleri",
|
||||||
"showOptionsPanel": "Yan Paneli Göster (O ya da T)",
|
|
||||||
"shuffle": "Kar",
|
"shuffle": "Kar",
|
||||||
"usePrompt": "İstemi Kullan",
|
"usePrompt": "İstemi Kullan",
|
||||||
"setToOptimalSizeTooSmall": "$t(parameters.setToOptimalSize) (çok küçük olabilir)",
|
"setToOptimalSizeTooSmall": "$t(parameters.setToOptimalSize) (çok küçük olabilir)",
|
||||||
@@ -346,7 +330,6 @@
|
|||||||
"perlinNoise": "Perlin Gürültüsü",
|
"perlinNoise": "Perlin Gürültüsü",
|
||||||
"scaledWidth": "Ölçekli En",
|
"scaledWidth": "Ölçekli En",
|
||||||
"seamlessXAxis": "Dikişsiz Döşeme X Ekseni",
|
"seamlessXAxis": "Dikişsiz Döşeme X Ekseni",
|
||||||
"downloadImage": "Görseli İndir",
|
|
||||||
"type": "Tür"
|
"type": "Tür"
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
@@ -399,11 +382,9 @@
|
|||||||
"defaultVAE": "Varsayılan VAE",
|
"defaultVAE": "Varsayılan VAE",
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
"noModelsAvailable": "Model yok",
|
"noModelsAvailable": "Model yok",
|
||||||
"noMatchingLoRAs": "Uygun LoRA Yok",
|
|
||||||
"noMatchingModels": "Uygun Model Yok",
|
"noMatchingModels": "Uygun Model Yok",
|
||||||
"loading": "yükleniyor",
|
"loading": "yükleniyor",
|
||||||
"selectModel": "Model Seçin",
|
"selectModel": "Model Seçin"
|
||||||
"noLoRAsInstalled": "LoRA Yok"
|
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"generation": "Oluşturma"
|
"generation": "Oluşturma"
|
||||||
@@ -411,7 +392,6 @@
|
|||||||
"sdxl": {
|
"sdxl": {
|
||||||
"cfgScale": "CFG Ölçeği",
|
"cfgScale": "CFG Ölçeği",
|
||||||
"loading": "Yükleniyor...",
|
"loading": "Yükleniyor...",
|
||||||
"denoisingStrength": "Arındırma Ölçüsü",
|
"denoisingStrength": "Arındırma Ölçüsü"
|
||||||
"concatPromptStyle": "İstem ve Stili Bitiştir"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,8 +22,7 @@
|
|||||||
"gallery": {
|
"gallery": {
|
||||||
"galleryImageSize": "Розмір зображень",
|
"galleryImageSize": "Розмір зображень",
|
||||||
"gallerySettings": "Налаштування галереї",
|
"gallerySettings": "Налаштування галереї",
|
||||||
"autoSwitchNewImages": "Автоматично вибирати нові",
|
"autoSwitchNewImages": "Автоматично вибирати нові"
|
||||||
"noImagesInGallery": "Зображень немає"
|
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
"modelManager": "Менеджер моделей",
|
"modelManager": "Менеджер моделей",
|
||||||
@@ -80,12 +79,10 @@
|
|||||||
"scaledHeight": "Масштаб В",
|
"scaledHeight": "Масштаб В",
|
||||||
"infillMethod": "Засіб заповнення",
|
"infillMethod": "Засіб заповнення",
|
||||||
"tileSize": "Розмір області",
|
"tileSize": "Розмір області",
|
||||||
"downloadImage": "Завантажити",
|
|
||||||
"usePrompt": "Використати запит",
|
"usePrompt": "Використати запит",
|
||||||
"useSeed": "Використати сід",
|
"useSeed": "Використати сід",
|
||||||
"useAll": "Використати все",
|
"useAll": "Використати все",
|
||||||
"info": "Метадані",
|
"info": "Метадані",
|
||||||
"showOptionsPanel": "Показати панель налаштувань",
|
|
||||||
"general": "Основне",
|
"general": "Основне",
|
||||||
"denoisingStrength": "Сила шумоподавлення",
|
"denoisingStrength": "Сила шумоподавлення",
|
||||||
"copyImage": "Копіювати зображення",
|
"copyImage": "Копіювати зображення",
|
||||||
|
|||||||
@@ -20,8 +20,6 @@
|
|||||||
"addBoard": "Thêm Bảng",
|
"addBoard": "Thêm Bảng",
|
||||||
"downloadBoard": "Tải Xuống Bảng",
|
"downloadBoard": "Tải Xuống Bảng",
|
||||||
"movingImagesToBoard_other": "Di chuyển {{count}} ảnh vào 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}}",
|
"noBoards": "Không Có Bảng Thuộc Loại {{boardType}}",
|
||||||
"noMatching": "Không Có Bảng Tương Ứng",
|
"noMatching": "Không Có Bảng Tương Ứng",
|
||||||
"searchBoard": "Tìm Bảng...",
|
"searchBoard": "Tìm Bảng...",
|
||||||
@@ -55,8 +53,12 @@
|
|||||||
"assetsWithCount_other": "{{count}} tài nguyên",
|
"assetsWithCount_other": "{{count}} tài nguyên",
|
||||||
"uncategorizedImages": "Ảnh Chưa Sắp Xếp",
|
"uncategorizedImages": "Ảnh Chưa Sắp Xếp",
|
||||||
"deleteAllUncategorizedImages": "Xoá Tất Cả Ả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": {
|
"gallery": {
|
||||||
"swapImages": "Đổi Hình Ảnh",
|
"swapImages": "Đổi Hình Ảnh",
|
||||||
@@ -84,33 +86,27 @@
|
|||||||
"galleryImageSize": "Kích Thước Ảnh",
|
"galleryImageSize": "Kích Thước Ảnh",
|
||||||
"downloadSelection": "Tải xuống Phần Được Lựa Chọn",
|
"downloadSelection": "Tải xuống Phần Được Lựa Chọn",
|
||||||
"bulkDownloadRequested": "Chuẩn Bị Tải Xuống",
|
"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",
|
"newestFirst": "Mới Nhất Trước",
|
||||||
"showStarredImagesFirst": "Hiển Thị Ảnh Gắn Sao 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.",
|
"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",
|
"starImage": "Gắn Sao",
|
||||||
"openViewer": "Mở Trình Xem",
|
|
||||||
"viewerImage": "Trình Xem Ảnh",
|
"viewerImage": "Trình Xem Ảnh",
|
||||||
"sideBySide": "Cạnh Nhau",
|
"sideBySide": "Cạnh Nhau",
|
||||||
"alwaysShowImageSizeBadge": "Luôn Hiển Thị Kích Thước Ảnh",
|
"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",
|
"autoAssignBoardOnClick": "Tự Động Gán Vào Bảng Khi Nhấp Chuột",
|
||||||
"jump": "Nhảy Đến",
|
|
||||||
"go": "Đi",
|
"go": "Đi",
|
||||||
"autoSwitchNewImages": "Tự Động Đổi Sang Hình Ảnh Mớ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.",
|
"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",
|
"openInViewer": "Mở Trong Trình Xem",
|
||||||
"searchImages": "Tìm Theo Metadata",
|
"searchImages": "Tìm Theo Metadata",
|
||||||
"selectForCompare": "Chọn Để So Sánh",
|
"selectForCompare": "Chọn Để So Sánh",
|
||||||
"closeViewer": "Đóng Trình Xem",
|
|
||||||
"move": "Di Chuyển",
|
"move": "Di Chuyển",
|
||||||
"displayBoardSearch": "Tìm Kiếm Bảng",
|
"displayBoardSearch": "Tìm Kiếm Bảng",
|
||||||
"displaySearch": "Tìm Kiếm Hình Ảnh",
|
"displaySearch": "Tìm Kiếm Hình Ảnh",
|
||||||
"selectAnImageToCompare": "Chọn Ảnh Để So Sánh",
|
|
||||||
"slider": "Thanh Trượt",
|
"slider": "Thanh Trượt",
|
||||||
"gallerySettings": "Cài Đặt Thư Viện Ảnh",
|
"gallerySettings": "Cài Đặt Thư Viện Ảnh",
|
||||||
"image": "hình ảnh",
|
"image": "hình ảnh",
|
||||||
"noImageSelected": "Không Có Ảnh Được Chọn",
|
"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.",
|
"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.",
|
"imagesTab": "Ảnh bạn vừa được tạo và lưu trong Invoke.",
|
||||||
"loading": "Đang Tải",
|
"loading": "Đang Tải",
|
||||||
@@ -118,13 +114,24 @@
|
|||||||
"exitCompare": "Ngừng So Sánh",
|
"exitCompare": "Ngừng So Sánh",
|
||||||
"stretchToFit": "Kéo Dài Cho Vừa Vặn",
|
"stretchToFit": "Kéo Dài Cho Vừa Vặn",
|
||||||
"sortDirection": "Cách Sắp Xếp",
|
"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.",
|
"compareHelp2": "Nhấn <Kbd>M</Kbd> để tuần hoàn trong chế độ so sánh.",
|
||||||
"boardsSettings": "Thiết Lập Bảng",
|
"boardsSettings": "Thiết Lập Bảng",
|
||||||
"imagesSettings": "Cài Đặt Ảnh Trong Thư Viện Ảnh",
|
"imagesSettings": "Cài Đặt Ảnh Trong Thư Viện Ảnh",
|
||||||
"assets": "Tài Nguyên",
|
"assets": "Tài Nguyên",
|
||||||
"images": "Hình Ảnh",
|
"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": {
|
"common": {
|
||||||
"ipAdapter": "IP Adapter",
|
"ipAdapter": "IP Adapter",
|
||||||
@@ -135,14 +142,12 @@
|
|||||||
"clipboard": "Clipboard",
|
"clipboard": "Clipboard",
|
||||||
"learnMore": "Tìm Hiểu Thêm",
|
"learnMore": "Tìm Hiểu Thêm",
|
||||||
"openInViewer": "Mở Trong Trình Xem",
|
"openInViewer": "Mở Trong Trình Xem",
|
||||||
"nextPage": "Trang Sau",
|
|
||||||
"alpha": "Alpha",
|
"alpha": "Alpha",
|
||||||
"edit": "Sửa",
|
"edit": "Sửa",
|
||||||
"nodes": "Workflow",
|
"nodes": "Workflow",
|
||||||
"format": "Định Dạng",
|
"format": "Định Dạng",
|
||||||
"delete": "Xoá",
|
"delete": "Xoá",
|
||||||
"details": "Chi Tiết",
|
"details": "Chi Tiết",
|
||||||
"imageFailedToLoad": "Không Thể Tải Hình Ảnh",
|
|
||||||
"img2img": "Hình ảnh sang Hình ảnh",
|
"img2img": "Hình ảnh sang Hình ảnh",
|
||||||
"upload": "Tải Lên",
|
"upload": "Tải Lên",
|
||||||
"somethingWentWrong": "Có vấn đề phát sinh",
|
"somethingWentWrong": "Có vấn đề phát sinh",
|
||||||
@@ -158,7 +163,7 @@
|
|||||||
"dontAskMeAgain": "Không hỏi lại",
|
"dontAskMeAgain": "Không hỏi lại",
|
||||||
"error": "Lỗi",
|
"error": "Lỗi",
|
||||||
"or": "hoặc",
|
"or": "hoặc",
|
||||||
"installed": "Đã Tải Xuống",
|
"installed": "Được Tải Xuống Sẵn",
|
||||||
"simple": "Cơ Bản",
|
"simple": "Cơ Bản",
|
||||||
"linear": "Tuyến Tính",
|
"linear": "Tuyến Tính",
|
||||||
"safetensors": "Safetensors",
|
"safetensors": "Safetensors",
|
||||||
@@ -180,19 +185,15 @@
|
|||||||
"on": "Bật",
|
"on": "Bật",
|
||||||
"checkpoint": "Checkpoint",
|
"checkpoint": "Checkpoint",
|
||||||
"txt2img": "Từ Ngữ Sang Hình Ảnh",
|
"txt2img": "Từ Ngữ Sang Hình Ảnh",
|
||||||
"prevPage": "Trang Trước",
|
|
||||||
"unknown": "Không Rõ",
|
"unknown": "Không Rõ",
|
||||||
"githubLabel": "Github",
|
"githubLabel": "Github",
|
||||||
"folder": "Thư mục",
|
"folder": "Thư mục",
|
||||||
"goTo": "Đến",
|
|
||||||
"hotkeysLabel": "Phím Tắt",
|
"hotkeysLabel": "Phím Tắt",
|
||||||
"loadingImage": "Đang Tải Hình ảnh",
|
"loadingImage": "Đang Tải Hình ảnh",
|
||||||
"localSystem": "Hệ Thống Máy Chủ",
|
|
||||||
"input": "Đầu Vào",
|
"input": "Đầu Vào",
|
||||||
"languagePickerLabel": "Ngôn Ngữ",
|
"languagePickerLabel": "Ngôn Ngữ",
|
||||||
"openInNewTab": "Mở Trong Tab Mới",
|
"openInNewTab": "Mở Trong Tab Mới",
|
||||||
"outpaint": "outpaint",
|
"outpaint": "outpaint",
|
||||||
"notInstalled": "Chưa $t(common.installed)",
|
|
||||||
"save": "Lưu",
|
"save": "Lưu",
|
||||||
"saveAs": "Lưu Như",
|
"saveAs": "Lưu Như",
|
||||||
"auto": "Tự Động",
|
"auto": "Tự Động",
|
||||||
@@ -234,7 +235,6 @@
|
|||||||
"end": "Kết Thúc",
|
"end": "Kết Thúc",
|
||||||
"min": "Tối Thiểu",
|
"min": "Tối Thiểu",
|
||||||
"max": "Tối Đa",
|
"max": "Tối Đa",
|
||||||
"resetToDefaults": "Đặt Lại Về Mặc Định",
|
|
||||||
"seed": "Hạt Giống",
|
"seed": "Hạt Giống",
|
||||||
"combinatorial": "Tổ Hợp",
|
"combinatorial": "Tổ Hợp",
|
||||||
"column": "Cột",
|
"column": "Cột",
|
||||||
@@ -256,7 +256,14 @@
|
|||||||
"options_withCount_other": "{{count}} thiết lập",
|
"options_withCount_other": "{{count}} thiết lập",
|
||||||
"removeNegativePrompt": "Xóa Lệnh Tiêu Cực",
|
"removeNegativePrompt": "Xóa Lệnh Tiêu Cực",
|
||||||
"addNegativePrompt": "Thêm 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": {
|
"prompt": {
|
||||||
"addPromptTrigger": "Thêm Trigger Cho Lệnh",
|
"addPromptTrigger": "Thêm Trigger Cho Lệnh",
|
||||||
@@ -266,11 +273,11 @@
|
|||||||
"expandCurrentPrompt": "Mở Rộng Lệnh Hiện Tại",
|
"expandCurrentPrompt": "Mở Rộng Lệnh Hiện Tại",
|
||||||
"uploadImageForPromptGeneration": "Tải Ảnh Để Tạo Sinh Lệnh",
|
"uploadImageForPromptGeneration": "Tải Ảnh Để Tạo Sinh Lệnh",
|
||||||
"expandingPrompt": "Đang mở rộng 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",
|
"resultTitle": "Mở Rộng Lệnh Hoàn Tất",
|
||||||
"resultSubtitle": "Chọn phương thức mở rộng lệnh:",
|
"resultSubtitle": "Chọn phương thức mở rộng lệnh:",
|
||||||
"replace": "Thay Thế",
|
"insert": "Chèn"
|
||||||
"insert": "Chèn",
|
|
||||||
"discard": "Huỷ Bỏ"
|
|
||||||
},
|
},
|
||||||
"queue": {
|
"queue": {
|
||||||
"resume": "Tiếp Tục",
|
"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?",
|
"clearQueueAlertDialog2": "Bạn chắc chắn muốn dọn sạch hàng không?",
|
||||||
"queueEmpty": "Hàng Trống",
|
"queueEmpty": "Hàng Trống",
|
||||||
"queueBack": "Thêm Vào Hàng",
|
"queueBack": "Thêm Vào Hàng",
|
||||||
"batchFieldValues": "Giá Trị Vùng Theo Lô",
|
|
||||||
"openQueue": "Mở Queue",
|
"openQueue": "Mở Queue",
|
||||||
"pause": "Dừng Lại",
|
"pause": "Dừng Lại",
|
||||||
"pauseFailed": "Có Vấn Đề Khi Dừng Lại Bộ Xử Lý",
|
"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",
|
"retryFailed": "Có Vấn Đề Khi Thử Lại Mục",
|
||||||
"retryItem": "Thử Lại Mục",
|
"retryItem": "Thử Lại Mục",
|
||||||
"credits": "Nguồn",
|
"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": {
|
"hotkeys": {
|
||||||
"canvas": {
|
"canvas": {
|
||||||
@@ -360,10 +372,6 @@
|
|||||||
"desc": "Phóng to canvas lên 800%.",
|
"desc": "Phóng to canvas lên 800%.",
|
||||||
"title": "Phóng To Vào 800%"
|
"title": "Phóng To Vào 800%"
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "Chỉnh Màu Sang Trắng",
|
|
||||||
"desc": "Chỉnh màu hiện tại sang màu trắng."
|
|
||||||
},
|
|
||||||
"transformSelected": {
|
"transformSelected": {
|
||||||
"title": "Biến Đổi",
|
"title": "Biến Đổi",
|
||||||
"desc": "Biến đổi layer được chọn."
|
"desc": "Biến đổi layer được chọn."
|
||||||
@@ -504,6 +512,14 @@
|
|||||||
"toggleBbox": {
|
"toggleBbox": {
|
||||||
"title": "Bật/Tắt Hiển Thị Hộp Giới Hạn",
|
"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"
|
"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": {
|
"workflows": {
|
||||||
@@ -701,12 +717,19 @@
|
|||||||
"title": "Chọn Tab Tạo Sinh",
|
"title": "Chọn Tab Tạo Sinh",
|
||||||
"desc": "Chọn tab Tạo Sinh.",
|
"desc": "Chọn tab Tạo Sinh.",
|
||||||
"key": "1"
|
"key": "1"
|
||||||
|
},
|
||||||
|
"selectVideoTab": {
|
||||||
|
"title": "Chọn Thẻ Video",
|
||||||
|
"desc": "Chọn thẻ Video."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"searchHotkeys": "Tìm Phím tắt",
|
"searchHotkeys": "Tìm Phím tắt",
|
||||||
"noHotkeysFound": "Không Tìm Thấy 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",
|
"clearSearch": "Làm Sạch Thanh Tìm Kiếm",
|
||||||
"hotkeys": "Phím Tắt"
|
"hotkeys": "Phím Tắt",
|
||||||
|
"video": {
|
||||||
|
"title": "Video"
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"modelManager": {
|
"modelManager": {
|
||||||
"modelConverted": "Model Đã Được Chuyển Đổi",
|
"modelConverted": "Model Đã Được Chuyển Đổi",
|
||||||
@@ -790,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.",
|
"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ỗ",
|
"inplaceInstall": "Tải Xuống Tại Chỗ",
|
||||||
"installRepo": "Tải Xuống Kho Lưu Trữ (Repository)",
|
"installRepo": "Tải Xuống Kho Lưu Trữ (Repository)",
|
||||||
"ipAdapters": "IP Adapters",
|
|
||||||
"loraModels": "LoRA",
|
"loraModels": "LoRA",
|
||||||
"main": "Chính",
|
"main": "Chính",
|
||||||
"modelConversionFailed": "Chuyển Đổi Model Thất Bại",
|
"modelConversionFailed": "Chuyển Đổi Model Thất Bại",
|
||||||
@@ -836,7 +858,6 @@
|
|||||||
"textualInversions": "Bộ Đảo Ngược Văn Bản",
|
"textualInversions": "Bộ Đảo Ngược Văn Bản",
|
||||||
"loraTriggerPhrases": "Từ Ngữ Kích Hoạt Cho LoRA",
|
"loraTriggerPhrases": "Từ Ngữ Kích Hoạt Cho LoRA",
|
||||||
"width": "Chiều Rộng",
|
"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",
|
"clipLEmbed": "CLIP-L Embed",
|
||||||
"clipGEmbed": "CLIP-G Embed",
|
"clipGEmbed": "CLIP-G Embed",
|
||||||
"controlLora": "LoRA Điều Khiển Được",
|
"controlLora": "LoRA Điều Khiển Được",
|
||||||
@@ -848,13 +869,11 @@
|
|||||||
"sigLip": "SigLIP",
|
"sigLip": "SigLIP",
|
||||||
"llavaOnevision": "LLaVA OneVision",
|
"llavaOnevision": "LLaVA OneVision",
|
||||||
"fileSize": "Kích Thước Tệp",
|
"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.",
|
"modelPickerFallbackNoModelsInstalled2": "Nhấp vào <LinkComponent>Trình Quản Lý Model</LinkComponent> để tải.",
|
||||||
"modelPickerFallbackNoModelsInstalled": "Không Có Sẵn Model.",
|
"modelPickerFallbackNoModelsInstalled": "Không Có Sẵn Model.",
|
||||||
"manageModels": "Quản Lý Model",
|
"manageModels": "Quản Lý Model",
|
||||||
"hfTokenReset": "Làm Mới HF Token",
|
"hfTokenReset": "Làm Mới HF Token",
|
||||||
"relatedModels": "Model Liên Quan",
|
"relatedModels": "Model Liên Quan",
|
||||||
"showOnlyRelatedModels": "Liên Quan",
|
|
||||||
"installedModelsCount": "Đã tải {{installed}} trên {{total}} model.",
|
"installedModelsCount": "Đã tải {{installed}} trên {{total}} model.",
|
||||||
"allNModelsInstalled": "Đã tải tất cả {{count}} model",
|
"allNModelsInstalled": "Đã tải tất cả {{count}} model",
|
||||||
"nToInstall": "Còn {{count}} để tải",
|
"nToInstall": "Còn {{count}} để tải",
|
||||||
@@ -871,30 +890,32 @@
|
|||||||
"scanFolderDescription": "Quét một thư mục trên máy để tự động tra và tải model.",
|
"scanFolderDescription": "Quét một thư mục trên máy để tự động tra và tải model.",
|
||||||
"recommendedModels": "Model Khuyến Nghị",
|
"recommendedModels": "Model Khuyến Nghị",
|
||||||
"exploreStarter": "Hoặc duyệt tất cả model khởi đầu có sẵn",
|
"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.",
|
"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:",
|
"browseAll": "Hoặc duyệt tất cả model có sẵn:",
|
||||||
"stableDiffusion15": "Stable Diffusion 1.5",
|
"stableDiffusion15": "Stable Diffusion 1.5",
|
||||||
"sdxl": "SDXL",
|
|
||||||
"fluxDev": "FLUX.1 dev"
|
"fluxDev": "FLUX.1 dev"
|
||||||
},
|
},
|
||||||
"installBundle": "Tải Xuống Gói",
|
"installBundle": "Tải Xuống Gói",
|
||||||
"installBundleMsg1": "Bạn có chắc chắn muốn tải xuống gói {{bundleName}}?",
|
"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": {
|
"metadata": {
|
||||||
"guidance": "Hướng Dẫn",
|
"guidance": "Hướng Dẫn",
|
||||||
"noRecallParameters": "Không tìm thấy tham số",
|
"noRecallParameters": "Không tìm thấy tham số",
|
||||||
"imageDetails": "Chi Tiết Ảnh",
|
"imageDetails": "Chi Tiết Ảnh",
|
||||||
"createdBy": "Được Tạo Bởi",
|
"createdBy": "Được Tạo Bởi",
|
||||||
"parsingFailed": "Lỗi Cú Pháp",
|
|
||||||
"canvasV2Metadata": "Layer Canvas",
|
"canvasV2Metadata": "Layer Canvas",
|
||||||
"parameterSet": "Dữ liệu tham số {{parameter}}",
|
"parameterSet": "Dữ liệu tham số {{parameter}}",
|
||||||
"positivePrompt": "Lệnh Tích Cực",
|
"positivePrompt": "Lệnh Tích Cực",
|
||||||
"recallParameter": "Gợi Nhớ {{label}}",
|
|
||||||
"seed": "Hạt Giống",
|
"seed": "Hạt Giống",
|
||||||
"negativePrompt": "Lệnh Tiêu Cực",
|
"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",
|
"strength": "Mức độ mạnh từ ảnh sang ảnh",
|
||||||
"Threshold": "Ngưỡng Nhiễu",
|
"Threshold": "Ngưỡng Nhiễu",
|
||||||
"width": "Chiều Rộng",
|
"width": "Chiều Rộng",
|
||||||
@@ -914,7 +935,15 @@
|
|||||||
"scheduler": "Scheduler",
|
"scheduler": "Scheduler",
|
||||||
"noMetaData": "Không tìm thấy metadata",
|
"noMetaData": "Không tìm thấy metadata",
|
||||||
"imageDimensions": "Kích Thước Ảnh",
|
"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": {
|
"accordions": {
|
||||||
"generation": {
|
"generation": {
|
||||||
@@ -960,8 +989,8 @@
|
|||||||
"method": "Cách Thức Sửa Độ Phân Giải Cao"
|
"method": "Cách Thức Sửa Độ Phân Giải Cao"
|
||||||
},
|
},
|
||||||
"hrf": "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",
|
"enableHrf": "Bật Chế Độ Chỉnh Sửa Phân Giải Cao",
|
||||||
"upscaleMethod": "Cách Thức Upscale"
|
"upscaleMethod": "Phương Thức Upscale"
|
||||||
},
|
},
|
||||||
"nodes": {
|
"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",
|
"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",
|
||||||
@@ -987,9 +1016,7 @@
|
|||||||
"float": "Số Thực",
|
"float": "Số Thực",
|
||||||
"missingNode": "Thiếu node kích hoạt",
|
"missingNode": "Thiếu node kích hoạt",
|
||||||
"currentImage": "Hình Ảnh Hiện Tại",
|
"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",
|
"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",
|
"workflowSettings": "Cài Đặt Biên Tập Workflow",
|
||||||
"workflowVersion": "Phiên Bản",
|
"workflowVersion": "Phiên Bản",
|
||||||
"unableToGetWorkflowVersion": "Không thể tìm phiên bản của lược đồ workflow",
|
"unableToGetWorkflowVersion": "Không thể tìm phiên bản của lược đồ workflow",
|
||||||
@@ -999,7 +1026,6 @@
|
|||||||
"ipAdapter": "IP Adapter",
|
"ipAdapter": "IP Adapter",
|
||||||
"cannotDuplicateConnection": "Không thể tạo hai kết nối trùng lặp",
|
"cannotDuplicateConnection": "Không thể tạo hai kết nối trùng lặp",
|
||||||
"workflowValidation": "Lỗi Xác Thực Workflow",
|
"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",
|
"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",
|
"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?)",
|
"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?)",
|
||||||
@@ -1013,7 +1039,6 @@
|
|||||||
"edge": "Kết Nối",
|
"edge": "Kết Nối",
|
||||||
"graph": "Đồ Thị",
|
"graph": "Đồ Thị",
|
||||||
"workflowAuthor": "Tác Giả",
|
"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",
|
"showEdgeLabels": "Hiển Thị Tên Kết Nối",
|
||||||
"unknownField": "Vùng Dữ Liệu Không Rõ",
|
"unknownField": "Vùng Dữ Liệu Không Rõ",
|
||||||
"executionStateCompleted": "Đã Hoàn Tất",
|
"executionStateCompleted": "Đã Hoàn Tất",
|
||||||
@@ -1043,7 +1068,6 @@
|
|||||||
"node": "Node",
|
"node": "Node",
|
||||||
"nodeTemplate": "Mẫu Trình Bày Của Node",
|
"nodeTemplate": "Mẫu Trình Bày Của Node",
|
||||||
"nodeType": "Loại 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ú",
|
"notes": "Ghi Chú",
|
||||||
"updateApp": "Cập Nhật Ứng Dụng",
|
"updateApp": "Cập Nhật Ứng Dụng",
|
||||||
"updateAllNodes": "Cập Nhật Các Node",
|
"updateAllNodes": "Cập Nhật Các Node",
|
||||||
@@ -1051,7 +1075,6 @@
|
|||||||
"imageAccessError": "Không thể tìm thấy ảnh {{image_name}}, chuyển về mặc định",
|
"imageAccessError": "Không thể tìm thấy ảnh {{image_name}}, chuyển về mặc định",
|
||||||
"unknownNode": "Node Không Rõ",
|
"unknownNode": "Node Không Rõ",
|
||||||
"unknownNodeType": "Loại 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",
|
"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ó",
|
"cannotConnectToSelf": "Không thể kết nối với chính nó",
|
||||||
"workflow": "Workflow",
|
"workflow": "Workflow",
|
||||||
@@ -1067,7 +1090,6 @@
|
|||||||
"fitViewportNodes": "Chế Độ Xem Vừa Khớp",
|
"fitViewportNodes": "Chế Độ Xem Vừa Khớp",
|
||||||
"fullyContainNodes": "Bao Phủ Node Hoàn Toàn Để Chọn",
|
"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",
|
"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ỏ",
|
"hideMinimapnodes": "Ẩn Bản Đồ Thu Nhỏ",
|
||||||
"inputMayOnlyHaveOneConnection": "Đầu vào chỉ có thể có một kết nối",
|
"inputMayOnlyHaveOneConnection": "Đầu vào chỉ có thể có một kết nối",
|
||||||
"noWorkflows": "Không Có Workflow",
|
"noWorkflows": "Không Có Workflow",
|
||||||
@@ -1078,34 +1100,27 @@
|
|||||||
"problemSettingTitle": "Có Vấn Đề Khi Thiết Lập Tiêu Đề",
|
"problemSettingTitle": "Có Vấn Đề Khi Thiết Lập Tiêu Đề",
|
||||||
"resetToDefaultValue": "Đặt lại giá trị mặc định",
|
"resetToDefaultValue": "Đặt lại giá trị mặc định",
|
||||||
"reloadNodeTemplates": "Tải Lại Mẫu Trình Bày Node",
|
"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",
|
"viewMode": "Dùng Chế Độ Xem Tuyến Tính",
|
||||||
"newWorkflowDesc": "Tạo workflow mới?",
|
"newWorkflowDesc": "Tạo workflow mới?",
|
||||||
"string": "Chuỗi Ký Tự",
|
"string": "Chuỗi Ký Tự",
|
||||||
"version": "Phiên Bản",
|
"version": "Phiên Bản",
|
||||||
"versionUnknown": " Phiên Bản Không Rõ",
|
|
||||||
"workflowContact": "Thông Tin Liên Lạc",
|
"workflowContact": "Thông Tin Liên Lạc",
|
||||||
"workflowName": "Tên",
|
"workflowName": "Tên",
|
||||||
"saveToGallery": "Lưu Vào Thư Viện Ảnh",
|
"saveToGallery": "Lưu Vào Thư Viện Ảnh",
|
||||||
"connectionWouldCreateCycle": "Kết nối này sẽ tạo ra vòng lặp",
|
"connectionWouldCreateCycle": "Kết nối này sẽ tạo ra vòng lặp",
|
||||||
"addNode": "Thêm Node",
|
"addNode": "Thêm Node",
|
||||||
"unsupportedAnyOfLength": "quá nhiều dữ liệu hợp nhất: {{count}}",
|
"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ị",
|
"validateConnections": "Xác Thực Kết Nối Và Đồ Thị",
|
||||||
"workflowNotes": "Ghi Chú",
|
"workflowNotes": "Ghi Chú",
|
||||||
"workflowTags": "Nhãn",
|
"workflowTags": "Nhãn",
|
||||||
"editMode": "Chỉnh sửa trong Trình Biên Tập Workflow",
|
"editMode": "Chỉnh sửa trong Trình Biên Tập Workflow",
|
||||||
"edit": "Chỉnh Sửa",
|
"edit": "Chỉnh Sửa",
|
||||||
"executionStateInProgress": "Đang Xử Lý",
|
"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}})",
|
"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",
|
"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.",
|
"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.",
|
"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",
|
"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",
|
"linearDistribution": "Phân Bố Tuyến Tính",
|
||||||
"uniformRandomDistribution": "Phân Bố Ngẫu Nhiên Đồng Nhất",
|
"uniformRandomDistribution": "Phân Bố Ngẫu Nhiên Đồng Nhất",
|
||||||
"parseString": "Phân Tích Chuỗi",
|
"parseString": "Phân Tích Chuỗi",
|
||||||
@@ -1114,7 +1129,6 @@
|
|||||||
"splitOn": "Tách Ở",
|
"splitOn": "Tách Ở",
|
||||||
"arithmeticSequence": "Cấp Số Cộng",
|
"arithmeticSequence": "Cấp Số Cộng",
|
||||||
"generatorNRandomValues_other": "{{count}} giá trị ngẫu nhiên",
|
"generatorNRandomValues_other": "{{count}} giá trị ngẫu nhiên",
|
||||||
"generatorLoading": "đang tải",
|
|
||||||
"generatorLoadFromFile": "Tải Từ Tệp",
|
"generatorLoadFromFile": "Tải Từ Tệp",
|
||||||
"dynamicPromptsRandom": "Dynamic Prompts (Ngẫu Nhiên)",
|
"dynamicPromptsRandom": "Dynamic Prompts (Ngẫu Nhiên)",
|
||||||
"dynamicPromptsCombinatorial": "Dynamic Prompts (Tổ Hợp)",
|
"dynamicPromptsCombinatorial": "Dynamic Prompts (Tổ Hợp)",
|
||||||
@@ -1124,7 +1138,6 @@
|
|||||||
"description": "Mô Tả",
|
"description": "Mô Tả",
|
||||||
"loadWorkflowDesc": "Tải workflow?",
|
"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.",
|
"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",
|
"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)",
|
"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",
|
"downloadWorkflowError": "Lỗi tải xuống workflow",
|
||||||
@@ -1150,7 +1163,23 @@
|
|||||||
"alignmentDL": "Dưới Cùng Bên Trái",
|
"alignmentDL": "Dưới Cùng Bên Trái",
|
||||||
"alignmentUR": "Trên Cùng Bên Phải",
|
"alignmentUR": "Trên Cùng Bên Phải",
|
||||||
"alignmentDR": "Dưới 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": {
|
"popovers": {
|
||||||
"paramCFGRescaleMultiplier": {
|
"paramCFGRescaleMultiplier": {
|
||||||
@@ -1598,14 +1627,14 @@
|
|||||||
"concepts": "LoRA",
|
"concepts": "LoRA",
|
||||||
"loading": "đang tải",
|
"loading": "đang tải",
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
"noMatchingLoRAs": "Không có LoRA phù hợp",
|
|
||||||
"noRefinerModelsInstalled": "Chưa có model SDXL Refiner được tải xuống",
|
"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",
|
"defaultVAE": "VAE Mặc Định",
|
||||||
"noMatchingModels": "Không có Model phù hợp",
|
"noMatchingModels": "Không có Model phù hợp",
|
||||||
"noModelsAvailable": "Không có model",
|
"noModelsAvailable": "Không có model",
|
||||||
"selectModel": "Chọn 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": {
|
"parameters": {
|
||||||
"postProcessing": "Xử Lý Hậu Kỳ (Shift + U)",
|
"postProcessing": "Xử Lý Hậu Kỳ (Shift + U)",
|
||||||
@@ -1615,9 +1644,7 @@
|
|||||||
"processImage": "Xử Lý Hình Ảnh",
|
"processImage": "Xử Lý Hình Ảnh",
|
||||||
"useSize": "Dùng Kích Thước",
|
"useSize": "Dùng Kích Thước",
|
||||||
"invoke": {
|
"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",
|
"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)",
|
"canvasIsFiltering": "Canvas đang bận (đang lọc)",
|
||||||
"canvasIsRasterizing": "Canvas đang bận (đang raster hoá)",
|
"canvasIsRasterizing": "Canvas đang bận (đang raster hoá)",
|
||||||
"canvasIsTransforming": "Canvas đang bận (đang biến đổi)",
|
"canvasIsTransforming": "Canvas đang bận (đang biến đổi)",
|
||||||
@@ -1631,8 +1658,6 @@
|
|||||||
"systemDisconnected": "Hệ thống mất kết nối",
|
"systemDisconnected": "Hệ thống mất kết nối",
|
||||||
"invoke": "Kích Hoạt",
|
"invoke": "Kích Hoạt",
|
||||||
"missingNodeTemplate": "Thiếu mẫu trình bày node",
|
"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",
|
"missingInputForField": "thiếu đầu vào",
|
||||||
"missingFieldTemplate": "Thiếu vùng mẫu trình bày",
|
"missingFieldTemplate": "Thiếu vùng mẫu trình bày",
|
||||||
"collectionTooFewItems": "quá ít mục, tối thiểu là {{minItems}}",
|
"collectionTooFewItems": "quá ít mục, tối thiểu là {{minItems}}",
|
||||||
@@ -1647,7 +1672,6 @@
|
|||||||
"collectionNumberLTExclusiveMin": "{{value}} <= {{exclusiveMinimum}} (giá trị chọn lọc tối thiểu)",
|
"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)",
|
"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}}",
|
"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}}",
|
"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",
|
"batchNodeEmptyCollection": "Một vài node hàng loạt có tài nguyên rỗng",
|
||||||
"collectionEmpty": "tài nguyên trống",
|
"collectionEmpty": "tài nguyên trống",
|
||||||
@@ -1658,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}}",
|
"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.",
|
"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",
|
"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",
|
"cfgScale": "Thang CFG",
|
||||||
"useSeed": "Dùng Hạt Giống",
|
"useSeed": "Dùng Hạt Giống",
|
||||||
@@ -1705,7 +1737,6 @@
|
|||||||
"useAll": "Dùng Tất Cả",
|
"useAll": "Dùng Tất Cả",
|
||||||
"useCpuNoise": "Dùng Độ Nhiễu CPU",
|
"useCpuNoise": "Dùng Độ Nhiễu CPU",
|
||||||
"remixImage": "Phối Lại Hình Ảnh",
|
"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",
|
"shuffle": "Xáo Trộn",
|
||||||
"setToOptimalSizeTooLarge": "$t(parameters.setToOptimalSize) (lớn quá)",
|
"setToOptimalSizeTooLarge": "$t(parameters.setToOptimalSize) (lớn quá)",
|
||||||
"cfgRescaleMultiplier": "Hệ Số Nhân Thang CFG",
|
"cfgRescaleMultiplier": "Hệ Số Nhân Thang CFG",
|
||||||
@@ -1715,7 +1746,6 @@
|
|||||||
"lockAspectRatio": "Khoá Tỉ Lệ",
|
"lockAspectRatio": "Khoá Tỉ Lệ",
|
||||||
"swapDimensions": "Hoán Đổi Kích Thước",
|
"swapDimensions": "Hoán Đổi Kích Thước",
|
||||||
"copyImage": "Sao Chép Hình Ảnh",
|
"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",
|
"imageFit": "Căn Chỉnh Ảnh Ban Đầu Thành Kích Thước Đầu Ra",
|
||||||
"info": "Thông Tin",
|
"info": "Thông Tin",
|
||||||
"usePrompt": "Dùng Lệnh",
|
"usePrompt": "Dùng Lệnh",
|
||||||
@@ -1723,7 +1753,17 @@
|
|||||||
"tileSize": "Kích Thước Khối",
|
"tileSize": "Kích Thước Khối",
|
||||||
"disabledNoRasterContent": "Đã Tắt (Không Có Nội Dung Dạng Raster)",
|
"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.",
|
"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": {
|
"dynamicPrompts": {
|
||||||
"seedBehaviour": {
|
"seedBehaviour": {
|
||||||
@@ -1749,9 +1789,7 @@
|
|||||||
"antialiasProgressImages": "Xử Lý Khử Răng Cưa Hình Ảnh",
|
"antialiasProgressImages": "Xử Lý Khử Răng Cưa Hình Ảnh",
|
||||||
"models": "Models",
|
"models": "Models",
|
||||||
"informationalPopoversDisabledDesc": "Hộp thoại hỗ trợ thông tin đã tắt. Bật lại trong Cài đặt.",
|
"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ả",
|
"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",
|
"enableNSFWChecker": "Bật Trình Kiểm Tra NSFW",
|
||||||
"clearIntermediatesWithCount_other": "Dọn sạch {{count}} sản phẩm trung gian",
|
"clearIntermediatesWithCount_other": "Dọn sạch {{count}} sản phẩm trung gian",
|
||||||
"reloadingIn": "Tải lại trong",
|
"reloadingIn": "Tải lại trong",
|
||||||
@@ -1774,7 +1812,9 @@
|
|||||||
"intermediatesClearedFailed": "Có Vấn Đề Khi Dọn Sạch Sản Phẩm Trung Gian",
|
"intermediatesClearedFailed": "Có Vấn Đề Khi Dọn Sạch Sản Phẩm Trung Gian",
|
||||||
"enableInvisibleWatermark": "Bật Chế Độ Ẩn Watermark",
|
"enableInvisibleWatermark": "Bật Chế Độ Ẩn Watermark",
|
||||||
"showDetailedInvocationProgress": "Hiện Dữ Liệu Xử Lý",
|
"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": {
|
"sdxl": {
|
||||||
"loading": "Đang Tải...",
|
"loading": "Đang Tải...",
|
||||||
@@ -1784,15 +1824,15 @@
|
|||||||
"refinermodel": "Model Refiner",
|
"refinermodel": "Model Refiner",
|
||||||
"refinerStart": "Bắt Đầu Refiner",
|
"refinerStart": "Bắt Đầu Refiner",
|
||||||
"denoisingStrength": "Sức Mạnh Khử Nhiễu",
|
"denoisingStrength": "Sức Mạnh Khử Nhiễu",
|
||||||
"posStylePrompt": "Điểm Tích Cực Cho Lệnh Phong Cách",
|
|
||||||
"scheduler": "Scheduler",
|
"scheduler": "Scheduler",
|
||||||
"refiner": "Refiner",
|
"refiner": "Refiner",
|
||||||
"cfgScale": "Thang CFG",
|
"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",
|
"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": {
|
"controlLayers": {
|
||||||
"width": "Chiều Rộng",
|
"width": "Chiều Rộng",
|
||||||
@@ -1807,7 +1847,6 @@
|
|||||||
"saveLayerToAssets": "Lưu Layer Vào Khu Tài Nguyên",
|
"saveLayerToAssets": "Lưu Layer Vào Khu Tài Nguyên",
|
||||||
"canvas": "Canvas",
|
"canvas": "Canvas",
|
||||||
"savedToGalleryOk": "Đã Lưu Vào Thư Viện Ảnh",
|
"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",
|
"clipToBbox": "Chuyển Nét Thành Hộp Giới Hạn",
|
||||||
"moveToFront": "Chuyển Lên Trước",
|
"moveToFront": "Chuyển Lên Trước",
|
||||||
"mergeVisible": "Gộp Layer Đang Hiển Thị",
|
"mergeVisible": "Gộp Layer Đang Hiển Thị",
|
||||||
@@ -1852,7 +1891,6 @@
|
|||||||
"stylePrecise": "Phong Cách (Chính Xác)",
|
"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."
|
"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",
|
"rasterLayer": "Layer Dạng Raster",
|
||||||
"disableAutoNegative": "Tắt Tự Động Đảo Chiều",
|
"disableAutoNegative": "Tắt Tự Động Đảo Chiều",
|
||||||
"controlLayer": "Layer Điều Khiển Được",
|
"controlLayer": "Layer Điều Khiển Được",
|
||||||
@@ -1863,8 +1901,6 @@
|
|||||||
"replaceLayer": "Thay Đổi Layer",
|
"replaceLayer": "Thay Đổi Layer",
|
||||||
"regionalGuidance": "Chỉ Dẫn Khu Vực",
|
"regionalGuidance": "Chỉ Dẫn Khu Vực",
|
||||||
"newCanvasFromImage": "Canvas Mới Từ Ảnh",
|
"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",
|
"convertRasterLayerTo": "Chuyển Đổi $t(controlLayers.rasterLayer) Thành",
|
||||||
"convertControlLayerTo": "Chuyển Đổi $t(controlLayers.controlLayer) Thành",
|
"convertControlLayerTo": "Chuyển Đổi $t(controlLayers.controlLayer) Thành",
|
||||||
"convertInpaintMaskTo": "Chuyển Đổi $t(controlLayers.inpaintMask) Thành",
|
"convertInpaintMaskTo": "Chuyển Đổi $t(controlLayers.inpaintMask) Thành",
|
||||||
@@ -1875,12 +1911,7 @@
|
|||||||
"newRasterLayer": "$t(controlLayers.rasterLayer) Mới",
|
"newRasterLayer": "$t(controlLayers.rasterLayer) Mới",
|
||||||
"enableAutoNegative": "Bật Tự Động Đảo Chiều",
|
"enableAutoNegative": "Bật Tự Động Đảo Chiều",
|
||||||
"sendToCanvas": "Chuyển Tới Canvas",
|
"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}}",
|
"hidingType": "Ẩn {{type}}",
|
||||||
"newImg2ImgCanvasFromImage": "Chuyển Đổi Ảnh Sang Ảnh Mới Từ Ảnh",
|
|
||||||
"copyToClipboard": "Sao Chép Vào Clipboard",
|
"copyToClipboard": "Sao Chép Vào Clipboard",
|
||||||
"logDebugInfo": "Thông Tin Log Gỡ Lỗi",
|
"logDebugInfo": "Thông Tin Log Gỡ Lỗi",
|
||||||
"regionalReferenceImage": "Ảnh Mẫu Khu Vực",
|
"regionalReferenceImage": "Ảnh Mẫu Khu Vực",
|
||||||
@@ -1893,37 +1924,28 @@
|
|||||||
"horizontal": "Đường Ngang",
|
"horizontal": "Đường Ngang",
|
||||||
"crosshatch": "Đường Chéo Song Song (Crosshatch)",
|
"crosshatch": "Đường Chéo Song Song (Crosshatch)",
|
||||||
"vertical": "Đường Dọc",
|
"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)",
|
"addControlLayer": "Thêm $t(controlLayers.controlLayer)",
|
||||||
"inpaintMask": "Lớp Phủ Inpaint",
|
"inpaintMask": "Lớp Phủ Inpaint",
|
||||||
"dynamicGrid": "Lưới Dynamic",
|
"dynamicGrid": "Lưới Dynamic",
|
||||||
"layer_other": "Layer",
|
"layer_other": "Layer",
|
||||||
"layer_withCount_other": "Layer ({{count}})",
|
|
||||||
"pullBboxIntoLayer": "Chuyển Hộp Giới Hạn Vào Layer",
|
"pullBboxIntoLayer": "Chuyển Hộp Giới Hạn Vào Layer",
|
||||||
"addInpaintMask": "Thêm $t(controlLayers.inpaintMask)",
|
"addInpaintMask": "Thêm $t(controlLayers.inpaintMask)",
|
||||||
"addRegionalGuidance": "Thêm $t(controlLayers.regionalGuidance)",
|
"addRegionalGuidance": "Thêm $t(controlLayers.regionalGuidance)",
|
||||||
"sendToGallery": "Đã Chuyển Tới Thư Viện Ảnh",
|
|
||||||
"unlocked": "Mở Khoá",
|
"unlocked": "Mở Khoá",
|
||||||
"addReferenceImage": "Thêm $t(controlLayers.referenceImage)",
|
"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",
|
"inpaintMask_withCount_other": "Lớp Phủ Inpaint",
|
||||||
"regionalGuidance_withCount_other": "Chỉ Dẫn Khu Vực",
|
"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",
|
"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",
|
"copyRasterLayerTo": "Sao Chép $t(controlLayers.rasterLayer) Tới",
|
||||||
"copyControlLayerTo": "Sao Chép $t(controlLayers.controlLayer) Tới",
|
"copyControlLayerTo": "Sao Chép $t(controlLayers.controlLayer) Tới",
|
||||||
"newRegionalGuidance": "$t(controlLayers.regionalGuidance) Mớ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",
|
"pullBboxIntoReferenceImage": "Chuyển Hộp Giới Hạn Vào Ảnh Mẫu",
|
||||||
"maskFill": "Lấp Đầy Lớp Phủ",
|
"maskFill": "Lấp Đầy Lớp Phủ",
|
||||||
"addRasterLayer": "Thêm $t(controlLayers.rasterLayer)",
|
"addRasterLayer": "Thêm $t(controlLayers.rasterLayer)",
|
||||||
"rasterLayers_withCount_hidden": "Layer Dạng Raster ({{count}} đang ẩn)",
|
|
||||||
"referenceImage": "Ảnh Mẫu",
|
"referenceImage": "Ảnh Mẫu",
|
||||||
"showProgressOnCanvas": "Hiện Quá Trình Xử Lý Lên Canvas",
|
"showProgressOnCanvas": "Hiện Quá Trình Xử Lý Lên Canvas",
|
||||||
"prompt": "Lệnh",
|
"prompt": "Lệnh",
|
||||||
@@ -1938,34 +1960,23 @@
|
|||||||
},
|
},
|
||||||
"addPositivePrompt": "Thêm $t(controlLayers.prompt)",
|
"addPositivePrompt": "Thêm $t(controlLayers.prompt)",
|
||||||
"deleteReferenceImage": "Xoá Ảnh Mẫu",
|
"deleteReferenceImage": "Xoá Ảnh Mẫu",
|
||||||
"inpaintMasks_withCount_visible": "Lớp Phủ Inpaint ({{count}})",
|
|
||||||
"disableTransparencyEffect": "Tắt Hiệu Ứng Trong Suốt",
|
"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",
|
"opacity": "Độ Mờ Đục",
|
||||||
"rectangle": "Hình Chữ Nhật",
|
"rectangle": "Hình Chữ Nhật",
|
||||||
"addNegativePrompt": "Thêm $t(controlLayers.negativePrompt)",
|
"addNegativePrompt": "Thêm $t(controlLayers.negativePrompt)",
|
||||||
"globalReferenceImage": "Ảnh Mẫu Toàn Vùng",
|
"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",
|
"controlLayer_withCount_other": "Layer Điều Khiển Được",
|
||||||
"newInpaintMask": "$t(controlLayers.inpaintMask) Mới",
|
"newInpaintMask": "$t(controlLayers.inpaintMask) Mới",
|
||||||
"locked": "Khoá",
|
"locked": "Khoá",
|
||||||
"newCanvasSession": "Phiên Canvas Mới",
|
|
||||||
"transparency": "Độ Trong Suốt",
|
"transparency": "Độ Trong Suốt",
|
||||||
"showingType": "Hiển Thị {{type}}",
|
"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": {
|
"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",
|
"invertSelection": "Đảo Ngược Phần Chọn",
|
||||||
"include": "Bao Gồm",
|
"include": "Bao Gồm",
|
||||||
"exclude": "Loại Trừ",
|
"exclude": "Loại Trừ",
|
||||||
"reset": "Làm Mới",
|
"reset": "Làm Mới",
|
||||||
"saveAs": "Lưu Như",
|
"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ó",
|
"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",
|
"clickToAdd": "Nhấp chuột vào layer để thêm điểm",
|
||||||
"clickToRemove": "Nhấp chuột vào một điểm để xoá",
|
"clickToRemove": "Nhấp chuột vào một điểm để xoá",
|
||||||
"selectObject": "Chọn Đối Tượng",
|
"selectObject": "Chọn Đối Tượng",
|
||||||
@@ -2199,7 +2210,6 @@
|
|||||||
"newSession": "Phiên Làm Việc Mới",
|
"newSession": "Phiên Làm Việc Mới",
|
||||||
"resetGenerationSettings": "Khởi Động Lại Cài Đặt Tạo Sinh",
|
"resetGenerationSettings": "Khởi Động Lại Cài Đặt Tạo Sinh",
|
||||||
"referenceImageRegional": "Ảnh Mẫu (Khu Vực)",
|
"referenceImageRegional": "Ảnh Mẫu (Khu Vực)",
|
||||||
"referenceImageGlobal": "Ảnh Mẫu (Toàn Vùng)",
|
|
||||||
"warnings": {
|
"warnings": {
|
||||||
"problemsFound": "Phát hiện vấn đề",
|
"problemsFound": "Phát hiện vấn đề",
|
||||||
"unsupportedModel": "layer không được hỗ trợ cho model cơ sở này",
|
"unsupportedModel": "layer không được hỗ trợ cho model cơ sở này",
|
||||||
@@ -2224,7 +2234,6 @@
|
|||||||
"pasteToBboxDesc": "Layer Mới (Trong Hộp Giới Hạn)",
|
"pasteToBboxDesc": "Layer Mới (Trong Hộp Giới Hạn)",
|
||||||
"pasteToCanvas": "Canvas",
|
"pasteToCanvas": "Canvas",
|
||||||
"pasteToCanvasDesc": "Layer Mới (Trong Canvas)",
|
"pasteToCanvasDesc": "Layer Mới (Trong Canvas)",
|
||||||
"pastedTo": "Dán Vào {{destination}}",
|
|
||||||
"regionCopiedToClipboard": "Sao Chép {{region}} Vào Clipboard",
|
"regionCopiedToClipboard": "Sao Chép {{region}} Vào Clipboard",
|
||||||
"copyRegionError": "Lỗi khi sao chép {{region}}",
|
"copyRegionError": "Lỗi khi sao chép {{region}}",
|
||||||
"errors": {
|
"errors": {
|
||||||
@@ -2244,7 +2253,6 @@
|
|||||||
"denoiseLimit": "Giới Hạn Khử Nhiễu",
|
"denoiseLimit": "Giới Hạn Khử Nhiễu",
|
||||||
"addImageNoise": "Thêm $t(controlLayers.imageNoise)",
|
"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.",
|
"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",
|
"exportCanvasToPSD": "Xuất Canvas Thành File PSD",
|
||||||
"ruleOfThirds": "Hiển Thị Quy Tắc Một Phần Ba",
|
"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)",
|
"showNonRasterLayers": "Hiển Thị Layer Không Thuộc Dạng Raster (Shift + H)",
|
||||||
@@ -2257,7 +2265,38 @@
|
|||||||
"fitBboxToMasks": "Xếp Vừa Hộp Giới Hạn Vào Lớp Phủ",
|
"fitBboxToMasks": "Xếp Vừa Hộp Giới Hạn Vào Lớp Phủ",
|
||||||
"invertMask": "Đảo Ngược Lớp Phủ",
|
"invertMask": "Đảo Ngược Lớp Phủ",
|
||||||
"maxRefImages": "Ảnh Mẫu Tối Đa",
|
"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": {
|
"stylePresets": {
|
||||||
"negativePrompt": "Lệnh Tiêu Cực",
|
"negativePrompt": "Lệnh Tiêu Cực",
|
||||||
@@ -2335,15 +2374,11 @@
|
|||||||
"toast": {
|
"toast": {
|
||||||
"imageUploadFailed": "Tải Lên Ảnh Thất Bại",
|
"imageUploadFailed": "Tải Lên Ảnh Thất Bại",
|
||||||
"layerCopiedToClipboard": "Sao Chép Layer Vào Clipboard",
|
"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",
|
"imageCopied": "Ảnh Đã Được Sao Chép",
|
||||||
"sentToUpscale": "Chuyển Vào Upscale",
|
"sentToUpscale": "Chuyển Vào Upscale",
|
||||||
"unableToLoadImage": "Không Thể Tải Hình Ảnh",
|
"unableToLoadImage": "Không Thể Tải Hình Ảnh",
|
||||||
"unableToLoadStylePreset": "Không Thể Tải Phong Cách Được Cài Đặt Trước",
|
"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",
|
"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",
|
"unableToLoadImageMetadata": "Không Thể Tải Metadata Của Ảnh",
|
||||||
"workflowLoaded": "Workflow Đã Tải",
|
"workflowLoaded": "Workflow Đã Tải",
|
||||||
"uploadFailed": "Tải Lên Thất Bại",
|
"uploadFailed": "Tải Lên Thất Bại",
|
||||||
@@ -2355,17 +2390,14 @@
|
|||||||
"importFailed": "Nhập Vào Thất Bại",
|
"importFailed": "Nhập Vào Thất Bại",
|
||||||
"importSuccessful": "Nhập Vào Thành Công",
|
"importSuccessful": "Nhập Vào Thành Công",
|
||||||
"workflowDeleted": "Workflow Đã Xoá",
|
"workflowDeleted": "Workflow Đã Xoá",
|
||||||
"setControlImage": "Đặt làm ảnh điều khiển được",
|
|
||||||
"connected": "Kết Nối Đến Server",
|
"connected": "Kết Nối Đến Server",
|
||||||
"imageUploaded": "Ảnh Đã Được Tải Lên",
|
"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",
|
"modelImportCanceled": "Nhập Vào Model Thất Bại",
|
||||||
"parameters": "Tham Số",
|
"parameters": "Tham Số",
|
||||||
"parameterSet": "Gợi Lại Tham Số",
|
"parameterSet": "Gợi Lại Tham Số",
|
||||||
"parameterSetDesc": "Gợi lại {{parameter}}",
|
"parameterSetDesc": "Gợi lại {{parameter}}",
|
||||||
"loadedWithWarnings": "Đã Tải Workflow Với Cảnh Báo",
|
"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.",
|
"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",
|
"problemRetrievingWorkflow": "Có Vấn Đề Khi Lấy Lại Workflow",
|
||||||
"somethingWentWrong": "Có Vấn Đề Phát Sinh",
|
"somethingWentWrong": "Có Vấn Đề Phát Sinh",
|
||||||
"problemDeletingWorkflow": "Có Vấn Đề Khi Xoá Workflow",
|
"problemDeletingWorkflow": "Có Vấn Đề Khi Xoá Workflow",
|
||||||
@@ -2375,13 +2407,12 @@
|
|||||||
"errorCopied": "Lỗi Khi Sao Chép",
|
"errorCopied": "Lỗi Khi Sao Chép",
|
||||||
"prunedQueue": "Cắt Bớt Hàng Đợi",
|
"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}}.",
|
"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ỷ",
|
"canceled": "Quá Trình Xử Lý Đã Huỷ",
|
||||||
"baseModelChanged": "Model Cơ Sở Đã Đổi",
|
"baseModelChanged": "Model Cơ Sở Đã Đổi",
|
||||||
"addedToUncategorized": "Thêm vào tài nguyên của bảng $t(boards.uncategorized)",
|
"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",
|
"linkCopied": "Đường Liên Kết Đã Được Sao Chép",
|
||||||
"outOfMemoryError": "Lỗi Vượt Quá Bộ Nhớ",
|
"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",
|
"modelAddedSimple": "Đã Thêm Model Vào Hàng Đợi",
|
||||||
"parametersSet": "Tham Số Đã Được Gợi Lại",
|
"parametersSet": "Tham Số Đã Được Gợi Lại",
|
||||||
"parameterNotSetDesc": "Không thể gợi lại {{parameter}}",
|
"parameterNotSetDesc": "Không thể gợi lại {{parameter}}",
|
||||||
@@ -2402,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.",
|
"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.",
|
"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.",
|
"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ị",
|
"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",
|
"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",
|
"invalidCanvasDimensions": "Kích Thước Canvas Không Phù Hợp",
|
||||||
"canvasTooLarge": "Canvas Quá Lớn",
|
"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.",
|
"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",
|
"psdExportSuccess": "Xuất File PSD Hoàn Tất",
|
||||||
"psdExportSuccessDesc": "Thành công xuất {{count}} layer sang file PSD",
|
"psdExportSuccessDesc": "Thành công xuất {{count}} layer sang file PSD",
|
||||||
"problemExportingPSD": "Có Vấn Đề Khi Xuất File PSD",
|
"problemExportingPSD": "Có Vấn Đề Khi Xuất File PSD",
|
||||||
"canvasManagerNotAvailable": "Trình Quản Lý Canvas Không Có Sẵn",
|
"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",
|
"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",
|
"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.",
|
"promptExpansionFailed": "Có vấn đề xảy ra. Hãy thử mở rộng lệnh lại.",
|
||||||
@@ -2424,6 +2449,20 @@
|
|||||||
"maskInvertFailed": "Thất Bại Khi Đảo Ngược Lớp Phủ",
|
"maskInvertFailed": "Thất Bại Khi Đảo Ngược Lớp Phủ",
|
||||||
"noVisibleMasks": "Không Có Lớp Phủ Đang Hiển Thị",
|
"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",
|
"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",
|
"noInpaintMaskSelected": "Không Có Lớp Phủ Inpant Được Chọn",
|
||||||
"noInpaintMaskSelectedDesc": "Chọn một lớp phủ inpaint để đảo ngược",
|
"noInpaintMaskSelectedDesc": "Chọn một lớp phủ inpaint để đảo ngược",
|
||||||
"invalidBbox": "Hộp Giới Hạn Không Hợp Lệ",
|
"invalidBbox": "Hộp Giới Hạn Không Hợp Lệ",
|
||||||
@@ -2440,7 +2479,8 @@
|
|||||||
"queue": "Queue (Hàng Đợi)",
|
"queue": "Queue (Hàng Đợi)",
|
||||||
"workflows": "Workflow (Luồng Làm Việc)",
|
"workflows": "Workflow (Luồng Làm Việc)",
|
||||||
"workflowsTab": "$t(common.tab) $t(ui.tabs.workflows)",
|
"workflowsTab": "$t(common.tab) $t(ui.tabs.workflows)",
|
||||||
"generate": "Tạo Sinh"
|
"generate": "Tạo Sinh",
|
||||||
|
"video": "Video"
|
||||||
},
|
},
|
||||||
"launchpad": {
|
"launchpad": {
|
||||||
"workflowsTitle": "Đi sâu hơn với Workflow.",
|
"workflowsTitle": "Đi sâu hơn với Workflow.",
|
||||||
@@ -2518,13 +2558,23 @@
|
|||||||
"generate": {
|
"generate": {
|
||||||
"canvasCalloutTitle": "Đang tìm cách để điều khiển, chỉnh sửa, và làm lại ảnh?",
|
"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."
|
"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": {
|
"panels": {
|
||||||
"launchpad": "Launchpad",
|
"launchpad": "Launchpad",
|
||||||
"workflowEditor": "Trình Biên Tập Workflow",
|
"workflowEditor": "Trình Biên Tập Workflow",
|
||||||
"imageViewer": "Trình Xem Ảnh",
|
"imageViewer": "Trình Xem",
|
||||||
"canvas": "Canvas"
|
"canvas": "Canvas",
|
||||||
|
"video": "Video"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"workflows": {
|
"workflows": {
|
||||||
@@ -2539,28 +2589,20 @@
|
|||||||
"saveWorkflowAs": "Lưu Workflow Như",
|
"saveWorkflowAs": "Lưu Workflow Như",
|
||||||
"downloadWorkflow": "Lưu Vào Tệp",
|
"downloadWorkflow": "Lưu Vào Tệp",
|
||||||
"noWorkflows": "Không Có Workflow",
|
"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...",
|
"savingWorkflow": "Đang Lưu Workflow...",
|
||||||
"ascending": "Tăng Dần",
|
"ascending": "Tăng Dần",
|
||||||
"loading": "Đang Tải Workflow",
|
"loading": "Đang Tải Workflow",
|
||||||
"chooseWorkflowFromLibrary": "Chọn Workflow Từ Thư Viện",
|
"chooseWorkflowFromLibrary": "Chọn Workflow Từ Thư Viện",
|
||||||
"workflows": "Workflow",
|
"workflows": "Workflow",
|
||||||
"copyShareLinkForWorkflow": "Sao Chép Liên Kết Chia Sẻ Cho Workflow",
|
"copyShareLinkForWorkflow": "Sao Chép Liên Kết Chia Sẻ Cho Workflow",
|
||||||
"openWorkflow": "Mở Workflow",
|
|
||||||
"name": "Tên",
|
"name": "Tên",
|
||||||
"unnamedWorkflow": "Workflow Vô Danh",
|
"unnamedWorkflow": "Workflow Vô Danh",
|
||||||
"saveWorkflow": "Lưu Workflow",
|
"saveWorkflow": "Lưu Workflow",
|
||||||
"problemSavingWorkflow": "Có Vấn Đề Khi Lưu Workflow",
|
"problemSavingWorkflow": "Có Vấn Đề Khi Lưu Workflow",
|
||||||
"noDescription": "Không có mô tả",
|
|
||||||
"updated": "Đã Cập Nhật",
|
"updated": "Đã Cập Nhật",
|
||||||
"uploadWorkflow": "Tải Từ Tệp",
|
"uploadWorkflow": "Tải Từ Tệp",
|
||||||
"autoLayout": "Bố Trí Tự Động",
|
"autoLayout": "Bố Trí Tự Động",
|
||||||
"loadWorkflow": "$t(common.load) Workflow",
|
"loadWorkflow": "$t(common.load) Workflow",
|
||||||
"searchWorkflows": "Tìm Workflow",
|
|
||||||
"newWorkflowCreated": "Workflow Mới Được Tạo",
|
"newWorkflowCreated": "Workflow Mới Được Tạo",
|
||||||
"workflowCleared": "Đã Dọn Dẹp Workflow",
|
"workflowCleared": "Đã Dọn Dẹp Workflow",
|
||||||
"loadFromGraph": "Tải Workflow Từ Đồ Thị",
|
"loadFromGraph": "Tải Workflow Từ Đồ Thị",
|
||||||
@@ -2571,7 +2613,6 @@
|
|||||||
"opened": "Đã Mở",
|
"opened": "Đã Mở",
|
||||||
"deleteWorkflow": "Xoá Workflow",
|
"deleteWorkflow": "Xoá Workflow",
|
||||||
"workflowEditorMenu": "Menu Biên Tập Workflow",
|
"workflowEditorMenu": "Menu Biên Tập Workflow",
|
||||||
"openLibrary": "Mở Thư Viện",
|
|
||||||
"builder": {
|
"builder": {
|
||||||
"resetAllNodeFields": "Tải Lại Các Vùng Node",
|
"resetAllNodeFields": "Tải Lại Các Vùng Node",
|
||||||
"builder": "Trình Tạo Vùng Nhập",
|
"builder": "Trình Tạo Vùng Nhập",
|
||||||
@@ -2587,7 +2628,6 @@
|
|||||||
"multiLine": "Nhiều Dòng",
|
"multiLine": "Nhiều Dòng",
|
||||||
"slider": "Thanh Trượt",
|
"slider": "Thanh Trượt",
|
||||||
"both": "Cả Hai",
|
"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.",
|
"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",
|
"containerPlaceholder": "Hộp Chứa Trống",
|
||||||
"headingPlaceholder": "Đầu Dòng Trống",
|
"headingPlaceholder": "Đầu Dòng Trống",
|
||||||
@@ -2596,7 +2636,6 @@
|
|||||||
"deleteAllElements": "Xóa Tất Cả Thành Phần",
|
"deleteAllElements": "Xóa Tất Cả Thành Phần",
|
||||||
"nodeField": "Vùng Node",
|
"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.",
|
"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",
|
"container": "Hộp Chứa",
|
||||||
"heading": "Đầu Dòng",
|
"heading": "Đầu Dòng",
|
||||||
"text": "Văn Bản",
|
"text": "Văn Bản",
|
||||||
@@ -2642,25 +2681,36 @@
|
|||||||
"errorWorkflowHasUnpublishableNodes": "Workflow có lô node, node sản sinh, hoặc node tách metadata",
|
"errorWorkflowHasUnpublishableNodes": "Workflow có lô node, node sản sinh, hoặc node tách metadata",
|
||||||
"removeFromForm": "Xóa Khỏi Vùng Nhập",
|
"removeFromForm": "Xóa Khỏi Vùng Nhập",
|
||||||
"showShuffle": "Hiện Xáo Trộn",
|
"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",
|
"yourWorkflows": "Workflow Của Bạn",
|
||||||
"browseWorkflows": "Khám Phá Workflow",
|
"browseWorkflows": "Khám Phá Workflow",
|
||||||
"workflowThumbnail": "Ảnh Minh Họa Workflow",
|
"workflowThumbnail": "Ảnh Minh Họa Workflow",
|
||||||
"saveChanges": "Lưu Thay Đổi",
|
"saveChanges": "Lưu Thay Đổi",
|
||||||
"allLoaded": "Đã Tải Tất Cả Workflow",
|
|
||||||
"shared": "Nhóm",
|
"shared": "Nhóm",
|
||||||
"searchPlaceholder": "Tìm theo tên, mô tả, hoặc nhãn",
|
"searchPlaceholder": "Tìm theo tên, mô tả, hoặc nhãn",
|
||||||
"filterByTags": "Lọc Theo Nhãn",
|
|
||||||
"recentlyOpened": "Mở Gần Đây",
|
"recentlyOpened": "Mở Gần Đây",
|
||||||
"private": "Cá Nhân",
|
"private": "Cá Nhân",
|
||||||
"loadMore": "Tải Thêm",
|
"loadMore": "Tải Thêm",
|
||||||
"view": "Xem",
|
"view": "Xem",
|
||||||
"deselectAll": "Huỷ Chọn Tất Cả",
|
"deselectAll": "Huỷ Chọn Tất Cả",
|
||||||
"noRecentWorkflows": "Không Có Workflows Gần Đây",
|
|
||||||
"recommended": "Có Thể Bạn Sẽ Cần",
|
"recommended": "Có Thể Bạn Sẽ Cần",
|
||||||
"emptyStringPlaceholder": "<xâu ký tự trống>",
|
"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": {
|
"upscaling": {
|
||||||
"missingUpscaleInitialImage": "Thiếu ảnh dùng để upscale",
|
"missingUpscaleInitialImage": "Thiếu ảnh dùng để upscale",
|
||||||
@@ -2697,11 +2747,11 @@
|
|||||||
"whatsNewInInvoke": "Có Gì Mới Ở Invoke",
|
"whatsNewInInvoke": "Có Gì Mới Ở Invoke",
|
||||||
"readReleaseNotes": "Đọc Ghi Chú Phát Hành",
|
"readReleaseNotes": "Đọc Ghi Chú Phát Hành",
|
||||||
"watchRecentReleaseVideos": "Xem Video Phát Hành Mới Nhất",
|
"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": [
|
"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",
|
"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'",
|
||||||
"Giảm lượng tiêu thụ VRAM cho các ảnh mẫu Kontext và mã hóa VAE"
|
"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": {
|
"upsell": {
|
||||||
"professional": "Chuyên Nghiệp",
|
"professional": "Chuyên Nghiệp",
|
||||||
@@ -2729,5 +2779,12 @@
|
|||||||
"clearSucceeded": "Cache Model Đã Được Dọn",
|
"clearSucceeded": "Cache Model Đã Được Dọn",
|
||||||
"clearFailed": "Có Vấn Đề Khi Dọn Cache Model",
|
"clearFailed": "Có Vấn Đề Khi Dọn Cache Model",
|
||||||
"clear": "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": "批次管理器",
|
"batch": "批次管理器",
|
||||||
"communityLabel": "社区",
|
"communityLabel": "社区",
|
||||||
"modelManager": "模型管理器",
|
"modelManager": "模型管理器",
|
||||||
"imageFailedToLoad": "无法加载图像",
|
|
||||||
"learnMore": "了解更多",
|
"learnMore": "了解更多",
|
||||||
"advanced": "高级",
|
"advanced": "高级",
|
||||||
"t2iAdapter": "T2I Adapter",
|
"t2iAdapter": "T2I Adapter",
|
||||||
@@ -51,23 +50,19 @@
|
|||||||
"somethingWentWrong": "出了点问题",
|
"somethingWentWrong": "出了点问题",
|
||||||
"copyError": "$t(gallery.copy) 错误",
|
"copyError": "$t(gallery.copy) 错误",
|
||||||
"input": "输入",
|
"input": "输入",
|
||||||
"notInstalled": "非 $t(common.installed)",
|
|
||||||
"delete": "删除",
|
"delete": "删除",
|
||||||
"updated": "已上传",
|
"updated": "已上传",
|
||||||
"save": "保存",
|
"save": "保存",
|
||||||
"created": "已创建",
|
"created": "已创建",
|
||||||
"prevPage": "上一页",
|
|
||||||
"unknownError": "未知错误",
|
"unknownError": "未知错误",
|
||||||
"direction": "指向",
|
"direction": "指向",
|
||||||
"orderBy": "排序方式:",
|
"orderBy": "排序方式:",
|
||||||
"nextPage": "下一页",
|
|
||||||
"saveAs": "保存为",
|
"saveAs": "保存为",
|
||||||
"ai": "ai",
|
"ai": "ai",
|
||||||
"or": "或",
|
"or": "或",
|
||||||
"aboutDesc": "使用 Invoke 工作?来看看:",
|
"aboutDesc": "使用 Invoke 工作?来看看:",
|
||||||
"add": "添加",
|
"add": "添加",
|
||||||
"copy": "复制",
|
"copy": "复制",
|
||||||
"localSystem": "本地系统",
|
|
||||||
"aboutHeading": "掌握你的创造力",
|
"aboutHeading": "掌握你的创造力",
|
||||||
"enabled": "已启用",
|
"enabled": "已启用",
|
||||||
"disabled": "已禁用",
|
"disabled": "已禁用",
|
||||||
@@ -78,7 +73,6 @@
|
|||||||
"selected": "选中的",
|
"selected": "选中的",
|
||||||
"green": "绿",
|
"green": "绿",
|
||||||
"blue": "蓝",
|
"blue": "蓝",
|
||||||
"goTo": "前往",
|
|
||||||
"dontShowMeThese": "请勿显示这些内容",
|
"dontShowMeThese": "请勿显示这些内容",
|
||||||
"beta": "测试版",
|
"beta": "测试版",
|
||||||
"toResolve": "解决",
|
"toResolve": "解决",
|
||||||
@@ -104,13 +98,11 @@
|
|||||||
"galleryImageSize": "预览大小",
|
"galleryImageSize": "预览大小",
|
||||||
"gallerySettings": "预览设置",
|
"gallerySettings": "预览设置",
|
||||||
"autoSwitchNewImages": "自动切换到新图像",
|
"autoSwitchNewImages": "自动切换到新图像",
|
||||||
"noImagesInGallery": "无图像可用于显示",
|
|
||||||
"deleteImage_other": "删除{{count}}张图片",
|
"deleteImage_other": "删除{{count}}张图片",
|
||||||
"deleteImagePermanent": "删除的图片无法被恢复。",
|
"deleteImagePermanent": "删除的图片无法被恢复。",
|
||||||
"autoAssignBoardOnClick": "点击后自动分配面板",
|
"autoAssignBoardOnClick": "点击后自动分配面板",
|
||||||
"featuresWillReset": "如果您删除该图像,这些功能会立即被重置。",
|
"featuresWillReset": "如果您删除该图像,这些功能会立即被重置。",
|
||||||
"loading": "加载中",
|
"loading": "加载中",
|
||||||
"unableToLoad": "无法加载图库",
|
|
||||||
"currentlyInUse": "该图像目前在以下功能中使用:",
|
"currentlyInUse": "该图像目前在以下功能中使用:",
|
||||||
"copy": "复制",
|
"copy": "复制",
|
||||||
"download": "下载",
|
"download": "下载",
|
||||||
@@ -125,7 +117,6 @@
|
|||||||
"starImage": "收藏图像",
|
"starImage": "收藏图像",
|
||||||
"alwaysShowImageSizeBadge": "始终显示图像尺寸",
|
"alwaysShowImageSizeBadge": "始终显示图像尺寸",
|
||||||
"selectForCompare": "选择以比较",
|
"selectForCompare": "选择以比较",
|
||||||
"selectAnImageToCompare": "选择一个图像进行比较",
|
|
||||||
"slider": "滑块",
|
"slider": "滑块",
|
||||||
"sideBySide": "并排",
|
"sideBySide": "并排",
|
||||||
"bulkDownloadFailed": "下载失败",
|
"bulkDownloadFailed": "下载失败",
|
||||||
@@ -148,7 +139,6 @@
|
|||||||
"newestFirst": "最新在前",
|
"newestFirst": "最新在前",
|
||||||
"compareHelp4": "按 <Kbd>Z</Kbd>或 <Kbd>Esc</Kbd> 键退出。",
|
"compareHelp4": "按 <Kbd>Z</Kbd>或 <Kbd>Esc</Kbd> 键退出。",
|
||||||
"searchImages": "按元数据搜索",
|
"searchImages": "按元数据搜索",
|
||||||
"jump": "跳过",
|
|
||||||
"compareHelp2": "按 <Kbd>M</Kbd> 键切换不同的比较模式。",
|
"compareHelp2": "按 <Kbd>M</Kbd> 键切换不同的比较模式。",
|
||||||
"displayBoardSearch": "板块搜索",
|
"displayBoardSearch": "板块搜索",
|
||||||
"displaySearch": "图像搜索",
|
"displaySearch": "图像搜索",
|
||||||
@@ -161,8 +151,6 @@
|
|||||||
"gallery": "画廊",
|
"gallery": "画廊",
|
||||||
"move": "移动",
|
"move": "移动",
|
||||||
"imagesTab": "您在Invoke中创建和保存的图片。",
|
"imagesTab": "您在Invoke中创建和保存的图片。",
|
||||||
"openViewer": "打开查看器",
|
|
||||||
"closeViewer": "关闭查看器",
|
|
||||||
"assetsTab": "您已上传用于项目的文件。"
|
"assetsTab": "您已上传用于项目的文件。"
|
||||||
},
|
},
|
||||||
"hotkeys": {
|
"hotkeys": {
|
||||||
@@ -310,10 +298,6 @@
|
|||||||
"title": "移动工具",
|
"title": "移动工具",
|
||||||
"desc": "选择移动工具。"
|
"desc": "选择移动工具。"
|
||||||
},
|
},
|
||||||
"setFillToWhite": {
|
|
||||||
"title": "将颜色设置为白色",
|
|
||||||
"desc": "将当前工具的颜色设置为白色。"
|
|
||||||
},
|
|
||||||
"cancelTransform": {
|
"cancelTransform": {
|
||||||
"desc": "取消待处理的变换。",
|
"desc": "取消待处理的变换。",
|
||||||
"title": "取消变换"
|
"title": "取消变换"
|
||||||
@@ -577,9 +561,7 @@
|
|||||||
"huggingFacePlaceholder": "所有者或模型名称",
|
"huggingFacePlaceholder": "所有者或模型名称",
|
||||||
"huggingFaceRepoID": "HuggingFace仓库ID",
|
"huggingFaceRepoID": "HuggingFace仓库ID",
|
||||||
"loraTriggerPhrases": "LoRA 触发词",
|
"loraTriggerPhrases": "LoRA 触发词",
|
||||||
"ipAdapters": "IP适配器",
|
|
||||||
"spandrelImageToImage": "图生图(Spandrel)",
|
"spandrelImageToImage": "图生图(Spandrel)",
|
||||||
"starterModelsInModelManager": "您可以在模型管理器中找到初始模型",
|
|
||||||
"noDefaultSettings": "此模型没有配置默认设置。请访问模型管理器添加默认设置。",
|
"noDefaultSettings": "此模型没有配置默认设置。请访问模型管理器添加默认设置。",
|
||||||
"clipEmbed": "CLIP 嵌入",
|
"clipEmbed": "CLIP 嵌入",
|
||||||
"defaultSettingsOutOfSync": "某些设置与模型的默认值不匹配:",
|
"defaultSettingsOutOfSync": "某些设置与模型的默认值不匹配:",
|
||||||
@@ -630,12 +612,10 @@
|
|||||||
"scaledHeight": "缩放长度",
|
"scaledHeight": "缩放长度",
|
||||||
"infillMethod": "填充方法",
|
"infillMethod": "填充方法",
|
||||||
"tileSize": "方格尺寸",
|
"tileSize": "方格尺寸",
|
||||||
"downloadImage": "下载图像",
|
|
||||||
"usePrompt": "使用提示",
|
"usePrompt": "使用提示",
|
||||||
"useSeed": "使用种子",
|
"useSeed": "使用种子",
|
||||||
"useAll": "使用所有参数",
|
"useAll": "使用所有参数",
|
||||||
"info": "信息",
|
"info": "信息",
|
||||||
"showOptionsPanel": "显示侧栏浮窗 (O 或 T)",
|
|
||||||
"seamlessYAxis": "无缝平铺 Y 轴",
|
"seamlessYAxis": "无缝平铺 Y 轴",
|
||||||
"seamlessXAxis": "无缝平铺 X 轴",
|
"seamlessXAxis": "无缝平铺 X 轴",
|
||||||
"denoisingStrength": "去噪强度",
|
"denoisingStrength": "去噪强度",
|
||||||
@@ -661,15 +641,11 @@
|
|||||||
"addingImagesTo": "添加图像到",
|
"addingImagesTo": "添加图像到",
|
||||||
"noPrompts": "没有已生成的提示词",
|
"noPrompts": "没有已生成的提示词",
|
||||||
"canvasIsFiltering": "画布正在过滤",
|
"canvasIsFiltering": "画布正在过滤",
|
||||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),缩放后的边界框高度为 {{height}}",
|
|
||||||
"noCLIPEmbedModelSelected": "未为FLUX生成选择CLIP嵌入模型",
|
"noCLIPEmbedModelSelected": "未为FLUX生成选择CLIP嵌入模型",
|
||||||
"noFLUXVAEModelSelected": "未为FLUX生成选择VAE模型",
|
"noFLUXVAEModelSelected": "未为FLUX生成选择VAE模型",
|
||||||
"canvasIsRasterizing": "画布正在栅格化",
|
"canvasIsRasterizing": "画布正在栅格化",
|
||||||
"canvasIsCompositing": "画布正在合成",
|
"canvasIsCompositing": "画布正在合成",
|
||||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),边界框宽度为 {{width}}",
|
|
||||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),缩放后的边界框宽度为 {{width}}",
|
|
||||||
"noT5EncoderModelSelected": "未为FLUX生成选择T5编码器模型",
|
"noT5EncoderModelSelected": "未为FLUX生成选择T5编码器模型",
|
||||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),边界框高度为 {{height}}",
|
|
||||||
"canvasIsTransforming": "画布正在变换"
|
"canvasIsTransforming": "画布正在变换"
|
||||||
},
|
},
|
||||||
"patchmatchDownScaleSize": "缩小",
|
"patchmatchDownScaleSize": "缩小",
|
||||||
@@ -733,8 +709,6 @@
|
|||||||
"informationalPopoversDisabledDesc": "信息提示框已被禁用.请在设置中重新启用.",
|
"informationalPopoversDisabledDesc": "信息提示框已被禁用.请在设置中重新启用.",
|
||||||
"enableModelDescriptions": "在下拉菜单中启用模型描述",
|
"enableModelDescriptions": "在下拉菜单中启用模型描述",
|
||||||
"confirmOnNewSession": "新会话时确认",
|
"confirmOnNewSession": "新会话时确认",
|
||||||
"modelDescriptionsDisabledDesc": "下拉菜单中的模型描述已被禁用。可在设置中启用。",
|
|
||||||
"modelDescriptionsDisabled": "下拉菜单中的模型描述已禁用",
|
|
||||||
"showDetailedInvocationProgress": "显示进度详情"
|
"showDetailedInvocationProgress": "显示进度详情"
|
||||||
},
|
},
|
||||||
"toast": {
|
"toast": {
|
||||||
@@ -750,14 +724,11 @@
|
|||||||
"problemCopyingImage": "无法复制图像",
|
"problemCopyingImage": "无法复制图像",
|
||||||
"modelAddedSimple": "模型已加入队列",
|
"modelAddedSimple": "模型已加入队列",
|
||||||
"loadedWithWarnings": "已加载带有警告的工作流",
|
"loadedWithWarnings": "已加载带有警告的工作流",
|
||||||
"setControlImage": "设为控制图像",
|
|
||||||
"setNodeField": "设为节点字段",
|
|
||||||
"imageUploaded": "图像已上传",
|
"imageUploaded": "图像已上传",
|
||||||
"addedToBoard": "添加到{{name}}的资产中",
|
"addedToBoard": "添加到{{name}}的资产中",
|
||||||
"workflowLoaded": "工作流已加载",
|
"workflowLoaded": "工作流已加载",
|
||||||
"imageUploadFailed": "图像上传失败",
|
"imageUploadFailed": "图像上传失败",
|
||||||
"baseModelChangedCleared_other": "已清除或禁用{{count}}个不兼容的子模型",
|
"baseModelChangedCleared_other": "已清除或禁用{{count}}个不兼容的子模型",
|
||||||
"invalidUpload": "无效的上传",
|
|
||||||
"problemDeletingWorkflow": "删除工作流时出现问题",
|
"problemDeletingWorkflow": "删除工作流时出现问题",
|
||||||
"workflowDeleted": "已删除工作流",
|
"workflowDeleted": "已删除工作流",
|
||||||
"problemRetrievingWorkflow": "检索工作流时发生问题",
|
"problemRetrievingWorkflow": "检索工作流时发生问题",
|
||||||
@@ -777,21 +748,16 @@
|
|||||||
"modelImportCanceled": "模型导入已取消",
|
"modelImportCanceled": "模型导入已取消",
|
||||||
"importFailed": "导入失败",
|
"importFailed": "导入失败",
|
||||||
"importSuccessful": "导入成功",
|
"importSuccessful": "导入成功",
|
||||||
"layerSavedToAssets": "图层已保存到资产",
|
|
||||||
"sentToUpscale": "已发送到放大处理",
|
"sentToUpscale": "已发送到放大处理",
|
||||||
"addedToUncategorized": "已添加到看板 $t(boards.uncategorized) 的资产中",
|
"addedToUncategorized": "已添加到看板 $t(boards.uncategorized) 的资产中",
|
||||||
"linkCopied": "链接已复制",
|
"linkCopied": "链接已复制",
|
||||||
"uploadFailedInvalidUploadDesc_withCount_other": "最多只能上传 {{count}} 张 PNG 或 JPEG 图像。",
|
|
||||||
"problemSavingLayer": "无法保存图层",
|
"problemSavingLayer": "无法保存图层",
|
||||||
"unableToLoadImage": "无法加载图像",
|
"unableToLoadImage": "无法加载图像",
|
||||||
"imageNotLoadedDesc": "无法找到图像",
|
|
||||||
"unableToLoadStylePreset": "无法加载样式预设",
|
"unableToLoadStylePreset": "无法加载样式预设",
|
||||||
"stylePresetLoaded": "样式预设已加载",
|
"stylePresetLoaded": "样式预设已加载",
|
||||||
"problemCopyingLayer": "无法复制图层",
|
"problemCopyingLayer": "无法复制图层",
|
||||||
"sentToCanvas": "已发送到画布",
|
"sentToCanvas": "已发送到画布",
|
||||||
"unableToLoadImageMetadata": "无法加载图像元数据",
|
"unableToLoadImageMetadata": "无法加载图像元数据",
|
||||||
"imageSaved": "图像已保存",
|
|
||||||
"imageSavingFailed": "图像保存失败",
|
|
||||||
"layerCopiedToClipboard": "图层已复制到剪贴板",
|
"layerCopiedToClipboard": "图层已复制到剪贴板",
|
||||||
"imagesWillBeAddedTo": "上传的图像将添加到看板 {{boardName}} 的资产中。"
|
"imagesWillBeAddedTo": "上传的图像将添加到看板 {{boardName}} 的资产中。"
|
||||||
},
|
},
|
||||||
@@ -819,11 +785,8 @@
|
|||||||
"fitViewportNodes": "自适应视图",
|
"fitViewportNodes": "自适应视图",
|
||||||
"showMinimapnodes": "显示缩略图",
|
"showMinimapnodes": "显示缩略图",
|
||||||
"hideMinimapnodes": "隐藏缩略图",
|
"hideMinimapnodes": "隐藏缩略图",
|
||||||
"showLegendNodes": "显示字段类型图例",
|
|
||||||
"hideLegendNodes": "隐藏字段类型图例",
|
|
||||||
"downloadWorkflow": "下载工作流 JSON",
|
"downloadWorkflow": "下载工作流 JSON",
|
||||||
"workflowDescription": "简述",
|
"workflowDescription": "简述",
|
||||||
"versionUnknown": " 未知版本",
|
|
||||||
"noNodeSelected": "无选中的节点",
|
"noNodeSelected": "无选中的节点",
|
||||||
"addNode": "添加节点",
|
"addNode": "添加节点",
|
||||||
"unableToValidateWorkflow": "无法验证工作流",
|
"unableToValidateWorkflow": "无法验证工作流",
|
||||||
@@ -833,9 +796,7 @@
|
|||||||
"workflowContact": "联系",
|
"workflowContact": "联系",
|
||||||
"animatedEdges": "边缘动效",
|
"animatedEdges": "边缘动效",
|
||||||
"nodeTemplate": "节点模板",
|
"nodeTemplate": "节点模板",
|
||||||
"unableToLoadWorkflow": "无法加载工作流",
|
|
||||||
"snapToGrid": "对齐网格",
|
"snapToGrid": "对齐网格",
|
||||||
"noFieldsLinearview": "线性视图中未添加任何字段",
|
|
||||||
"nodeSearch": "检索节点",
|
"nodeSearch": "检索节点",
|
||||||
"version": "版本",
|
"version": "版本",
|
||||||
"validateConnections": "验证连接和节点图",
|
"validateConnections": "验证连接和节点图",
|
||||||
@@ -850,8 +811,6 @@
|
|||||||
"fieldTypesMustMatch": "类型必须匹配",
|
"fieldTypesMustMatch": "类型必须匹配",
|
||||||
"workflow": "工作流",
|
"workflow": "工作流",
|
||||||
"animatedEdgesHelp": "为选中边缘和其连接的选中节点的边缘添加动画",
|
"animatedEdgesHelp": "为选中边缘和其连接的选中节点的边缘添加动画",
|
||||||
"unknownTemplate": "未知模板",
|
|
||||||
"removeLinearView": "从线性视图中移除",
|
|
||||||
"workflowTags": "标签",
|
"workflowTags": "标签",
|
||||||
"fullyContainNodesHelp": "节点必须完全位于选择框中才能被选中",
|
"fullyContainNodesHelp": "节点必须完全位于选择框中才能被选中",
|
||||||
"workflowValidation": "工作流验证错误",
|
"workflowValidation": "工作流验证错误",
|
||||||
@@ -885,7 +844,6 @@
|
|||||||
"node": "节点",
|
"node": "节点",
|
||||||
"collection": "合集",
|
"collection": "合集",
|
||||||
"string": "字符串",
|
"string": "字符串",
|
||||||
"mismatchedVersion": "无效的节点:类型为 {{type}} 的节点 {{node}} 版本不匹配(是否尝试更新?)",
|
|
||||||
"cannotDuplicateConnection": "无法创建重复的连接",
|
"cannotDuplicateConnection": "无法创建重复的连接",
|
||||||
"enum": "Enum (枚举)",
|
"enum": "Enum (枚举)",
|
||||||
"float": "浮点",
|
"float": "浮点",
|
||||||
@@ -896,7 +854,6 @@
|
|||||||
"unableToUpdateNodes_other": "{{count}} 个节点无法完成更新",
|
"unableToUpdateNodes_other": "{{count}} 个节点无法完成更新",
|
||||||
"inputFieldTypeParseError": "无法解析 {{node}} 的输入类型 {{field}}。({{message}})",
|
"inputFieldTypeParseError": "无法解析 {{node}} 的输入类型 {{field}}。({{message}})",
|
||||||
"unsupportedArrayItemType": "不支持的数组类型 \"{{type}}\"",
|
"unsupportedArrayItemType": "不支持的数组类型 \"{{type}}\"",
|
||||||
"addLinearView": "添加到线性视图",
|
|
||||||
"targetNodeFieldDoesNotExist": "无效的边缘:{{node}} 的目标/输入区域 {{field}} 不存在",
|
"targetNodeFieldDoesNotExist": "无效的边缘:{{node}} 的目标/输入区域 {{field}} 不存在",
|
||||||
"unsupportedMismatchedUnion": "合集或标量类型与基类 {{firstType}} 和 {{secondType}} 不匹配",
|
"unsupportedMismatchedUnion": "合集或标量类型与基类 {{firstType}} 和 {{secondType}} 不匹配",
|
||||||
"allNodesUpdated": "已更新所有节点",
|
"allNodesUpdated": "已更新所有节点",
|
||||||
@@ -916,7 +873,6 @@
|
|||||||
"collectionOrScalarFieldType": "{{name}} (单一项目或项目集合)",
|
"collectionOrScalarFieldType": "{{name}} (单一项目或项目集合)",
|
||||||
"nodeVersion": "节点版本",
|
"nodeVersion": "节点版本",
|
||||||
"deletedInvalidEdge": "已删除无效的边缘 {{source}} -> {{target}}",
|
"deletedInvalidEdge": "已删除无效的边缘 {{source}} -> {{target}}",
|
||||||
"unknownInput": "未知输入:{{name}}",
|
|
||||||
"prototypeDesc": "此调用是一个原型 (prototype)。它可能会在本项目更新期间发生破坏性更改,并且随时可能被删除。",
|
"prototypeDesc": "此调用是一个原型 (prototype)。它可能会在本项目更新期间发生破坏性更改,并且随时可能被删除。",
|
||||||
"betaDesc": "此调用尚处于测试阶段。在稳定之前,它可能会在项目更新期间发生破坏性更改。本项目计划长期支持这种调用。",
|
"betaDesc": "此调用尚处于测试阶段。在稳定之前,它可能会在项目更新期间发生破坏性更改。本项目计划长期支持这种调用。",
|
||||||
"newWorkflow": "新建工作流",
|
"newWorkflow": "新建工作流",
|
||||||
@@ -928,7 +884,6 @@
|
|||||||
"missingNode": "缺少调用节点",
|
"missingNode": "缺少调用节点",
|
||||||
"missingInvocationTemplate": "缺少调用模版",
|
"missingInvocationTemplate": "缺少调用模版",
|
||||||
"noFieldsViewMode": "此工作流程未选择任何要显示的字段.请查看完整工作流程以进行配置.",
|
"noFieldsViewMode": "此工作流程未选择任何要显示的字段.请查看完整工作流程以进行配置.",
|
||||||
"reorderLinearView": "调整线性视图顺序",
|
|
||||||
"viewMode": "在线性视图中使用",
|
"viewMode": "在线性视图中使用",
|
||||||
"showEdgeLabelsHelp": "在边缘上显示标签,指示连接的节点",
|
"showEdgeLabelsHelp": "在边缘上显示标签,指示连接的节点",
|
||||||
"cannotMixAndMatchCollectionItemTypes": "集合项目类型不能混用",
|
"cannotMixAndMatchCollectionItemTypes": "集合项目类型不能混用",
|
||||||
@@ -1002,7 +957,6 @@
|
|||||||
"session": "会话",
|
"session": "会话",
|
||||||
"enqueueing": "队列中的批次",
|
"enqueueing": "队列中的批次",
|
||||||
"graphFailedToQueue": "节点图加入队列失败",
|
"graphFailedToQueue": "节点图加入队列失败",
|
||||||
"batchFieldValues": "批处理值",
|
|
||||||
"time": "时间",
|
"time": "时间",
|
||||||
"openQueue": "打开队列",
|
"openQueue": "打开队列",
|
||||||
"prompts_other": "提示词",
|
"prompts_other": "提示词",
|
||||||
@@ -1021,18 +975,14 @@
|
|||||||
"refinerStart": "Refiner 开始作用时机",
|
"refinerStart": "Refiner 开始作用时机",
|
||||||
"scheduler": "调度器",
|
"scheduler": "调度器",
|
||||||
"cfgScale": "CFG 等级",
|
"cfgScale": "CFG 等级",
|
||||||
"negStylePrompt": "负向样式提示词",
|
|
||||||
"noModelsAvailable": "无可用模型",
|
"noModelsAvailable": "无可用模型",
|
||||||
"negAestheticScore": "负向美学评分",
|
"negAestheticScore": "负向美学评分",
|
||||||
"denoisingStrength": "去噪强度",
|
"denoisingStrength": "去噪强度",
|
||||||
"refinermodel": "Refiner 模型",
|
"refinermodel": "Refiner 模型",
|
||||||
"posAestheticScore": "正向美学评分",
|
"posAestheticScore": "正向美学评分",
|
||||||
"concatPromptStyle": "链接提示词 & 样式",
|
|
||||||
"loading": "加载中...",
|
"loading": "加载中...",
|
||||||
"steps": "步数",
|
"steps": "步数",
|
||||||
"posStylePrompt": "正向样式提示词",
|
|
||||||
"refiner": "Refiner",
|
"refiner": "Refiner",
|
||||||
"freePromptStyle": "手动输入样式提示词",
|
|
||||||
"refinerSteps": "精炼步数"
|
"refinerSteps": "精炼步数"
|
||||||
},
|
},
|
||||||
"metadata": {
|
"metadata": {
|
||||||
@@ -1059,8 +1009,6 @@
|
|||||||
"vae": "VAE",
|
"vae": "VAE",
|
||||||
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
||||||
"allPrompts": "所有提示",
|
"allPrompts": "所有提示",
|
||||||
"parsingFailed": "解析失败",
|
|
||||||
"recallParameter": "调用{{label}}",
|
|
||||||
"imageDimensions": "图像尺寸",
|
"imageDimensions": "图像尺寸",
|
||||||
"parameterSet": "已设置参数{{parameter}}",
|
"parameterSet": "已设置参数{{parameter}}",
|
||||||
"guidance": "指导",
|
"guidance": "指导",
|
||||||
@@ -1071,11 +1019,9 @@
|
|||||||
"models": {
|
"models": {
|
||||||
"noMatchingModels": "无相匹配的模型",
|
"noMatchingModels": "无相匹配的模型",
|
||||||
"loading": "加载中",
|
"loading": "加载中",
|
||||||
"noMatchingLoRAs": "无相匹配的 LoRA",
|
|
||||||
"noModelsAvailable": "无可用模型",
|
"noModelsAvailable": "无可用模型",
|
||||||
"selectModel": "选择一个模型",
|
"selectModel": "选择一个模型",
|
||||||
"noRefinerModelsInstalled": "无已安装的 SDXL Refiner 模型",
|
"noRefinerModelsInstalled": "无已安装的 SDXL Refiner 模型",
|
||||||
"noLoRAsInstalled": "无已安装的 LoRA",
|
|
||||||
"addLora": "添加 LoRA",
|
"addLora": "添加 LoRA",
|
||||||
"lora": "LoRA",
|
"lora": "LoRA",
|
||||||
"defaultVAE": "默认 VAE",
|
"defaultVAE": "默认 VAE",
|
||||||
@@ -1104,10 +1050,8 @@
|
|||||||
"deletedBoardsCannotbeRestored": "删除的面板无法恢复。选择“仅删除面板”选项后,相关图片将会被移至未分类区域。",
|
"deletedBoardsCannotbeRestored": "删除的面板无法恢复。选择“仅删除面板”选项后,相关图片将会被移至未分类区域。",
|
||||||
"movingImagesToBoard_other": "移动 {{count}} 张图像到面板:",
|
"movingImagesToBoard_other": "移动 {{count}} 张图像到面板:",
|
||||||
"selectedForAutoAdd": "已选中自动添加",
|
"selectedForAutoAdd": "已选中自动添加",
|
||||||
"hideBoards": "隐藏面板",
|
|
||||||
"noBoards": "没有{{boardType}}类型的面板",
|
"noBoards": "没有{{boardType}}类型的面板",
|
||||||
"unarchiveBoard": "恢复面板",
|
"unarchiveBoard": "恢复面板",
|
||||||
"viewBoards": "查看面板",
|
|
||||||
"addPrivateBoard": "创建私密面板",
|
"addPrivateBoard": "创建私密面板",
|
||||||
"addSharedBoard": "创建共享面板",
|
"addSharedBoard": "创建共享面板",
|
||||||
"boards": "面板",
|
"boards": "面板",
|
||||||
@@ -1576,8 +1520,6 @@
|
|||||||
"useCache": "使用缓存"
|
"useCache": "使用缓存"
|
||||||
},
|
},
|
||||||
"hrf": {
|
"hrf": {
|
||||||
"enableHrf": "启用高分辨率修复",
|
|
||||||
"upscaleMethod": "放大方法",
|
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"strength": "高分辨率修复强度",
|
"strength": "高分辨率修复强度",
|
||||||
"enabled": "高分辨率修复已启用",
|
"enabled": "高分辨率修复已启用",
|
||||||
@@ -1590,20 +1532,15 @@
|
|||||||
"workflowEditorMenu": "工作流编辑器菜单",
|
"workflowEditorMenu": "工作流编辑器菜单",
|
||||||
"workflowName": "工作流名称",
|
"workflowName": "工作流名称",
|
||||||
"saveWorkflow": "保存工作流",
|
"saveWorkflow": "保存工作流",
|
||||||
"openWorkflow": "打开工作流",
|
|
||||||
"clearWorkflowSearchFilter": "清除工作流检索过滤器",
|
|
||||||
"workflowLibrary": "工作流库",
|
"workflowLibrary": "工作流库",
|
||||||
"downloadWorkflow": "保存到文件",
|
"downloadWorkflow": "保存到文件",
|
||||||
"workflowSaved": "已保存工作流",
|
"workflowSaved": "已保存工作流",
|
||||||
"unnamedWorkflow": "未命名的工作流",
|
"unnamedWorkflow": "未命名的工作流",
|
||||||
"savingWorkflow": "保存工作流中...",
|
"savingWorkflow": "保存工作流中...",
|
||||||
"problemLoading": "加载工作流时出现问题",
|
|
||||||
"loading": "加载工作流中",
|
"loading": "加载工作流中",
|
||||||
"searchWorkflows": "检索工作流",
|
|
||||||
"problemSavingWorkflow": "保存工作流时出现问题",
|
"problemSavingWorkflow": "保存工作流时出现问题",
|
||||||
"deleteWorkflow": "删除工作流",
|
"deleteWorkflow": "删除工作流",
|
||||||
"workflows": "工作流",
|
"workflows": "工作流",
|
||||||
"noDescription": "无描述",
|
|
||||||
"uploadWorkflow": "从文件中加载",
|
"uploadWorkflow": "从文件中加载",
|
||||||
"newWorkflowCreated": "已创建新的工作流",
|
"newWorkflowCreated": "已创建新的工作流",
|
||||||
"name": "名称",
|
"name": "名称",
|
||||||
@@ -1623,9 +1560,6 @@
|
|||||||
"copyShareLinkForWorkflow": "复制工作流程的分享链接",
|
"copyShareLinkForWorkflow": "复制工作流程的分享链接",
|
||||||
"delete": "删除",
|
"delete": "删除",
|
||||||
"download": "下载",
|
"download": "下载",
|
||||||
"defaultWorkflows": "默认工作流程",
|
|
||||||
"userWorkflows": "用户工作流程",
|
|
||||||
"projectWorkflows": "项目工作流程",
|
|
||||||
"copyShareLink": "复制分享链接",
|
"copyShareLink": "复制分享链接",
|
||||||
"chooseWorkflowFromLibrary": "从库中选择工作流程",
|
"chooseWorkflowFromLibrary": "从库中选择工作流程",
|
||||||
"deleteWorkflow2": "您确定要删除此工作流程吗?此操作无法撤销。"
|
"deleteWorkflow2": "您确定要删除此工作流程吗?此操作无法撤销。"
|
||||||
@@ -1663,7 +1597,6 @@
|
|||||||
"moveToBack": "移动到后面",
|
"moveToBack": "移动到后面",
|
||||||
"moveToFront": "移动到前面",
|
"moveToFront": "移动到前面",
|
||||||
"addLayer": "添加层",
|
"addLayer": "添加层",
|
||||||
"deletePrompt": "删除提示词",
|
|
||||||
"addPositivePrompt": "添加 $t(controlLayers.prompt)",
|
"addPositivePrompt": "添加 $t(controlLayers.prompt)",
|
||||||
"addNegativePrompt": "添加 $t(controlLayers.negativePrompt)",
|
"addNegativePrompt": "添加 $t(controlLayers.negativePrompt)",
|
||||||
"rectangle": "矩形",
|
"rectangle": "矩形",
|
||||||
@@ -1687,7 +1620,6 @@
|
|||||||
"maskFill": "遮罩填充",
|
"maskFill": "遮罩填充",
|
||||||
"newCanvasFromImage": "从图像创建新画布",
|
"newCanvasFromImage": "从图像创建新画布",
|
||||||
"pullBboxIntoReferenceImageOk": "边界框已导入到参考图像",
|
"pullBboxIntoReferenceImageOk": "边界框已导入到参考图像",
|
||||||
"globalReferenceImage_withCount_other": "全局参考图像",
|
|
||||||
"addInpaintMask": "添加 $t(controlLayers.inpaintMask)",
|
"addInpaintMask": "添加 $t(controlLayers.inpaintMask)",
|
||||||
"referenceImage": "参考图像",
|
"referenceImage": "参考图像",
|
||||||
"globalReferenceImage": "全局参考图像",
|
"globalReferenceImage": "全局参考图像",
|
||||||
@@ -1696,14 +1628,10 @@
|
|||||||
"copyRasterLayerTo": "复制 $t(controlLayers.rasterLayer) 到",
|
"copyRasterLayerTo": "复制 $t(controlLayers.rasterLayer) 到",
|
||||||
"clearHistory": "清除历史记录",
|
"clearHistory": "清除历史记录",
|
||||||
"inpaintMask": "修复遮罩",
|
"inpaintMask": "修复遮罩",
|
||||||
"regionalGuidance_withCount_visible": "区域引导({{count}} 个)",
|
|
||||||
"inpaintMasks_withCount_hidden": "修复遮罩({{count}} 个已隐藏)",
|
|
||||||
"enableAutoNegative": "启用自动负面提示",
|
"enableAutoNegative": "启用自动负面提示",
|
||||||
"disableAutoNegative": "禁用自动负面提示",
|
"disableAutoNegative": "禁用自动负面提示",
|
||||||
"deleteReferenceImage": "删除参考图像",
|
"deleteReferenceImage": "删除参考图像",
|
||||||
"sendToCanvas": "发送到画布",
|
"sendToCanvas": "发送到画布",
|
||||||
"controlLayers_withCount_visible": "控制图层({{count}} 个)",
|
|
||||||
"rasterLayers_withCount_visible": "栅格图层({{count}} 个)",
|
|
||||||
"convertRegionalGuidanceTo": "将 $t(controlLayers.regionalGuidance) 转换为",
|
"convertRegionalGuidanceTo": "将 $t(controlLayers.regionalGuidance) 转换为",
|
||||||
"newInpaintMask": "新建 $t(controlLayers.inpaintMask)",
|
"newInpaintMask": "新建 $t(controlLayers.inpaintMask)",
|
||||||
"regionIsEmpty": "选定区域为空",
|
"regionIsEmpty": "选定区域为空",
|
||||||
@@ -1715,14 +1643,12 @@
|
|||||||
"addRasterLayer": "添加 $t(controlLayers.rasterLayer)",
|
"addRasterLayer": "添加 $t(controlLayers.rasterLayer)",
|
||||||
"newRasterLayerOk": "已创建栅格层",
|
"newRasterLayerOk": "已创建栅格层",
|
||||||
"newRasterLayerError": "创建栅格层时出现问题",
|
"newRasterLayerError": "创建栅格层时出现问题",
|
||||||
"inpaintMasks_withCount_visible": "修复遮罩({{count}} 个)",
|
|
||||||
"convertRasterLayerTo": "将 $t(controlLayers.rasterLayer) 转换为",
|
"convertRasterLayerTo": "将 $t(controlLayers.rasterLayer) 转换为",
|
||||||
"copyControlLayerTo": "复制 $t(controlLayers.controlLayer) 到",
|
"copyControlLayerTo": "复制 $t(controlLayers.controlLayer) 到",
|
||||||
"copyInpaintMaskTo": "复制 $t(controlLayers.inpaintMask) 到",
|
"copyInpaintMaskTo": "复制 $t(controlLayers.inpaintMask) 到",
|
||||||
"copyRegionalGuidanceTo": "复制 $t(controlLayers.regionalGuidance) 到",
|
"copyRegionalGuidanceTo": "复制 $t(controlLayers.regionalGuidance) 到",
|
||||||
"newRasterLayer": "新建 $t(controlLayers.rasterLayer)",
|
"newRasterLayer": "新建 $t(controlLayers.rasterLayer)",
|
||||||
"newControlLayer": "新建 $t(controlLayers.controlLayer)",
|
"newControlLayer": "新建 $t(controlLayers.controlLayer)",
|
||||||
"newImg2ImgCanvasFromImage": "从图像创建新的图生图",
|
|
||||||
"rasterLayer": "栅格层",
|
"rasterLayer": "栅格层",
|
||||||
"controlLayer": "控制层",
|
"controlLayer": "控制层",
|
||||||
"outputOnlyMaskedRegions": "仅输出生成的区域",
|
"outputOnlyMaskedRegions": "仅输出生成的区域",
|
||||||
@@ -1735,36 +1661,22 @@
|
|||||||
"bboxOverlay": "显示边界框覆盖层",
|
"bboxOverlay": "显示边界框覆盖层",
|
||||||
"clipToBbox": "将Clip限制到边界框",
|
"clipToBbox": "将Clip限制到边界框",
|
||||||
"width": "宽度",
|
"width": "宽度",
|
||||||
"addGlobalReferenceImage": "添加 $t(controlLayers.globalReferenceImage)",
|
|
||||||
"inpaintMask_withCount_other": "修复遮罩",
|
"inpaintMask_withCount_other": "修复遮罩",
|
||||||
"regionalGuidance_withCount_other": "区域引导",
|
"regionalGuidance_withCount_other": "区域引导",
|
||||||
"newRegionalReferenceImageError": "创建局部参考图像时出现问题",
|
"newRegionalReferenceImageError": "创建局部参考图像时出现问题",
|
||||||
"pullBboxIntoLayerError": "将边界框导入图层时出现问题",
|
"pullBboxIntoLayerError": "将边界框导入图层时出现问题",
|
||||||
"pullBboxIntoLayerOk": "边界框已导入到图层",
|
"pullBboxIntoLayerOk": "边界框已导入到图层",
|
||||||
"sendToCanvasDesc": "按下“Invoke”按钮会将您的工作进度暂存到画布上。",
|
|
||||||
"sendToGallery": "发送到图库",
|
|
||||||
"sendToGalleryDesc": "按下“Invoke”键会生成并保存一张唯一的图像到您的图库中。",
|
|
||||||
"rasterLayer_withCount_other": "栅格图层",
|
"rasterLayer_withCount_other": "栅格图层",
|
||||||
"mergeDown": "向下合并",
|
"mergeDown": "向下合并",
|
||||||
"clearCaches": "清除缓存",
|
"clearCaches": "清除缓存",
|
||||||
"recalculateRects": "重新计算矩形",
|
"recalculateRects": "重新计算矩形",
|
||||||
"duplicate": "复制",
|
"duplicate": "复制",
|
||||||
"regionalGuidance_withCount_hidden": "区域引导({{count}} 个已隐藏)",
|
|
||||||
"convertControlLayerTo": "将 $t(controlLayers.controlLayer) 转换为",
|
"convertControlLayerTo": "将 $t(controlLayers.controlLayer) 转换为",
|
||||||
"convertInpaintMaskTo": "将 $t(controlLayers.inpaintMask) 转换为",
|
"convertInpaintMaskTo": "将 $t(controlLayers.inpaintMask) 转换为",
|
||||||
"viewProgressInViewer": "在 <Btn>图像查看器</Btn> 中查看进度和输出结果。",
|
|
||||||
"viewProgressOnCanvas": "在 <Btn>画布</Btn> 上查看进度和暂存的输出内容。",
|
|
||||||
"sendingToGallery": "将生成内容发送到图库",
|
|
||||||
"copyToClipboard": "复制到剪贴板",
|
"copyToClipboard": "复制到剪贴板",
|
||||||
"controlLayer_withCount_other": "控制图层",
|
"controlLayer_withCount_other": "控制图层",
|
||||||
"sendingToCanvas": "在画布上准备生成",
|
|
||||||
"addReferenceImage": "添加 $t(controlLayers.referenceImage)",
|
"addReferenceImage": "添加 $t(controlLayers.referenceImage)",
|
||||||
"addRegionalGuidance": "添加 $t(controlLayers.regionalGuidance)",
|
"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": "启用透明效果",
|
"enableTransparencyEffect": "启用透明效果",
|
||||||
"disableTransparencyEffect": "禁用透明效果",
|
"disableTransparencyEffect": "禁用透明效果",
|
||||||
"hidingType": "隐藏 {{type}}",
|
"hidingType": "隐藏 {{type}}",
|
||||||
|
|||||||
@@ -19,7 +19,6 @@
|
|||||||
"folder": "資料夾",
|
"folder": "資料夾",
|
||||||
"installed": "已安裝",
|
"installed": "已安裝",
|
||||||
"accept": "接受",
|
"accept": "接受",
|
||||||
"goTo": "前往",
|
|
||||||
"input": "輸入",
|
"input": "輸入",
|
||||||
"random": "隨機",
|
"random": "隨機",
|
||||||
"selected": "已選擇",
|
"selected": "已選擇",
|
||||||
@@ -29,8 +28,7 @@
|
|||||||
"copy": "複製",
|
"copy": "複製",
|
||||||
"error": "錯誤",
|
"error": "錯誤",
|
||||||
"file": "檔案",
|
"file": "檔案",
|
||||||
"format": "格式",
|
"format": "格式"
|
||||||
"imageFailedToLoad": "無法載入圖片"
|
|
||||||
},
|
},
|
||||||
"accessibility": {
|
"accessibility": {
|
||||||
"invokeProgressBar": "Invoke 進度條",
|
"invokeProgressBar": "Invoke 進度條",
|
||||||
@@ -179,8 +177,7 @@
|
|||||||
"workflowAuthor": "作者",
|
"workflowAuthor": "作者",
|
||||||
"version": "版本",
|
"version": "版本",
|
||||||
"executionStateCompleted": "已完成",
|
"executionStateCompleted": "已完成",
|
||||||
"edge": "邊緣",
|
"edge": "邊緣"
|
||||||
"versionUnknown": " 版本未知"
|
|
||||||
},
|
},
|
||||||
"sdxl": {
|
"sdxl": {
|
||||||
"steps": "步數",
|
"steps": "步數",
|
||||||
|
|||||||
@@ -2,12 +2,12 @@ import { useAppSelector } from 'app/store/storeHooks';
|
|||||||
import { useIsRegionFocused } from 'common/hooks/focus';
|
import { useIsRegionFocused } from 'common/hooks/focus';
|
||||||
import { useAssertSingleton } from 'common/hooks/useAssertSingleton';
|
import { useAssertSingleton } from 'common/hooks/useAssertSingleton';
|
||||||
import { useLoadWorkflow } from 'features/gallery/hooks/useLoadWorkflow';
|
import { useLoadWorkflow } from 'features/gallery/hooks/useLoadWorkflow';
|
||||||
import { useRecallAll } from 'features/gallery/hooks/useRecallAll';
|
import { useRecallAll } from 'features/gallery/hooks/useRecallAllImageMetadata';
|
||||||
import { useRecallDimensions } from 'features/gallery/hooks/useRecallDimensions';
|
import { useRecallDimensions } from 'features/gallery/hooks/useRecallDimensions';
|
||||||
import { useRecallPrompts } from 'features/gallery/hooks/useRecallPrompts';
|
import { useRecallPrompts } from 'features/gallery/hooks/useRecallPrompts';
|
||||||
import { useRecallRemix } from 'features/gallery/hooks/useRecallRemix';
|
import { useRecallRemix } from 'features/gallery/hooks/useRecallRemix';
|
||||||
import { useRecallSeed } from 'features/gallery/hooks/useRecallSeed';
|
import { useRecallSeed } from 'features/gallery/hooks/useRecallSeed';
|
||||||
import { selectLastSelectedImage } from 'features/gallery/store/gallerySelectors';
|
import { selectLastSelectedItem } from 'features/gallery/store/gallerySelectors';
|
||||||
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||||
import { memo } from 'react';
|
import { memo } from 'react';
|
||||||
import { useImageDTO } from 'services/api/endpoints/images';
|
import { useImageDTO } from 'services/api/endpoints/images';
|
||||||
@@ -15,8 +15,8 @@ import type { ImageDTO } from 'services/api/types';
|
|||||||
|
|
||||||
export const GlobalImageHotkeys = memo(() => {
|
export const GlobalImageHotkeys = memo(() => {
|
||||||
useAssertSingleton('GlobalImageHotkeys');
|
useAssertSingleton('GlobalImageHotkeys');
|
||||||
const imageName = useAppSelector(selectLastSelectedImage);
|
const lastSelectedItem = useAppSelector(selectLastSelectedItem);
|
||||||
const imageDTO = useImageDTO(imageName);
|
const imageDTO = useImageDTO(lastSelectedItem?.type === 'image' ? lastSelectedItem.id : null);
|
||||||
|
|
||||||
if (!imageDTO) {
|
if (!imageDTO) {
|
||||||
return null;
|
return null;
|
||||||
|
|||||||
@@ -2,11 +2,14 @@ import { GlobalImageHotkeys } from 'app/components/GlobalImageHotkeys';
|
|||||||
import ChangeBoardModal from 'features/changeBoardModal/components/ChangeBoardModal';
|
import ChangeBoardModal from 'features/changeBoardModal/components/ChangeBoardModal';
|
||||||
import { CanvasPasteModal } from 'features/controlLayers/components/CanvasPasteModal';
|
import { CanvasPasteModal } from 'features/controlLayers/components/CanvasPasteModal';
|
||||||
import { CanvasManagerProviderGate } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
import { CanvasManagerProviderGate } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
||||||
|
import { CropImageModal } from 'features/cropper/components/CropImageModal';
|
||||||
import { DeleteImageModal } from 'features/deleteImageModal/components/DeleteImageModal';
|
import { DeleteImageModal } from 'features/deleteImageModal/components/DeleteImageModal';
|
||||||
|
import { DeleteVideoModal } from 'features/deleteVideoModal/components/DeleteVideoModal';
|
||||||
import { FullscreenDropzone } from 'features/dnd/FullscreenDropzone';
|
import { FullscreenDropzone } from 'features/dnd/FullscreenDropzone';
|
||||||
import { DynamicPromptsModal } from 'features/dynamicPrompts/components/DynamicPromptsPreviewModal';
|
import { DynamicPromptsModal } from 'features/dynamicPrompts/components/DynamicPromptsPreviewModal';
|
||||||
import DeleteBoardModal from 'features/gallery/components/Boards/DeleteBoardModal';
|
import DeleteBoardModal from 'features/gallery/components/Boards/DeleteBoardModal';
|
||||||
import { ImageContextMenu } from 'features/gallery/components/ImageContextMenu/ImageContextMenu';
|
import { ImageContextMenu } from 'features/gallery/components/ContextMenu/ImageContextMenu';
|
||||||
|
import { VideoContextMenu } from 'features/gallery/components/ContextMenu/VideoContextMenu';
|
||||||
import { ShareWorkflowModal } from 'features/nodes/components/sidePanel/workflow/WorkflowLibrary/ShareWorkflowModal';
|
import { ShareWorkflowModal } from 'features/nodes/components/sidePanel/workflow/WorkflowLibrary/ShareWorkflowModal';
|
||||||
import { WorkflowLibraryModal } from 'features/nodes/components/sidePanel/workflow/WorkflowLibrary/WorkflowLibraryModal';
|
import { WorkflowLibraryModal } from 'features/nodes/components/sidePanel/workflow/WorkflowLibrary/WorkflowLibraryModal';
|
||||||
import { CancelAllExceptCurrentQueueItemConfirmationAlertDialog } from 'features/queue/components/CancelAllExceptCurrentQueueItemConfirmationAlertDialog';
|
import { CancelAllExceptCurrentQueueItemConfirmationAlertDialog } from 'features/queue/components/CancelAllExceptCurrentQueueItemConfirmationAlertDialog';
|
||||||
@@ -31,6 +34,7 @@ export const GlobalModalIsolator = memo(() => {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<DeleteImageModal />
|
<DeleteImageModal />
|
||||||
|
<DeleteVideoModal />
|
||||||
<ChangeBoardModal />
|
<ChangeBoardModal />
|
||||||
<DynamicPromptsModal />
|
<DynamicPromptsModal />
|
||||||
<StylePresetModal />
|
<StylePresetModal />
|
||||||
@@ -47,6 +51,7 @@ export const GlobalModalIsolator = memo(() => {
|
|||||||
<DeleteBoardModal />
|
<DeleteBoardModal />
|
||||||
<GlobalImageHotkeys />
|
<GlobalImageHotkeys />
|
||||||
<ImageContextMenu />
|
<ImageContextMenu />
|
||||||
|
<VideoContextMenu />
|
||||||
<FullscreenDropzone />
|
<FullscreenDropzone />
|
||||||
<VideosModal />
|
<VideosModal />
|
||||||
<SaveWorkflowAsDialog />
|
<SaveWorkflowAsDialog />
|
||||||
@@ -54,6 +59,7 @@ export const GlobalModalIsolator = memo(() => {
|
|||||||
<CanvasPasteModal />
|
<CanvasPasteModal />
|
||||||
</CanvasManagerProviderGate>
|
</CanvasManagerProviderGate>
|
||||||
<LoadWorkflowFromGraphModal />
|
<LoadWorkflowFromGraphModal />
|
||||||
|
<CropImageModal />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,16 +1,14 @@
|
|||||||
import 'i18n';
|
import 'i18n';
|
||||||
|
|
||||||
import type { Middleware } from '@reduxjs/toolkit';
|
import type { InvokeAIUIProps } from 'app/components/types';
|
||||||
import type { StudioInitAction } from 'app/hooks/useStudioInitAction';
|
|
||||||
import { $didStudioInit } from 'app/hooks/useStudioInitAction';
|
import { $didStudioInit } from 'app/hooks/useStudioInitAction';
|
||||||
import type { LoggingOverrides } from 'app/logging/logger';
|
|
||||||
import { $loggingOverrides, configureLogging } from 'app/logging/logger';
|
import { $loggingOverrides, configureLogging } from 'app/logging/logger';
|
||||||
import { addStorageListeners } from 'app/store/enhancers/reduxRemember/driver';
|
import { addStorageListeners } from 'app/store/enhancers/reduxRemember/driver';
|
||||||
import { $accountSettingsLink } from 'app/store/nanostores/accountSettingsLink';
|
import { $accountSettingsLink } from 'app/store/nanostores/accountSettingsLink';
|
||||||
|
import { $accountTypeText } from 'app/store/nanostores/accountTypeText';
|
||||||
import { $authToken } from 'app/store/nanostores/authToken';
|
import { $authToken } from 'app/store/nanostores/authToken';
|
||||||
import { $baseUrl } from 'app/store/nanostores/baseUrl';
|
import { $baseUrl } from 'app/store/nanostores/baseUrl';
|
||||||
import { $customNavComponent } from 'app/store/nanostores/customNavComponent';
|
import { $customNavComponent } from 'app/store/nanostores/customNavComponent';
|
||||||
import type { CustomStarUi } from 'app/store/nanostores/customStarUI';
|
|
||||||
import { $customStarUI } from 'app/store/nanostores/customStarUI';
|
import { $customStarUI } from 'app/store/nanostores/customStarUI';
|
||||||
import { $isDebugging } from 'app/store/nanostores/isDebugging';
|
import { $isDebugging } from 'app/store/nanostores/isDebugging';
|
||||||
import { $logo } from 'app/store/nanostores/logo';
|
import { $logo } from 'app/store/nanostores/logo';
|
||||||
@@ -20,11 +18,10 @@ import { $projectId, $projectName, $projectUrl } from 'app/store/nanostores/proj
|
|||||||
import { $queueId, DEFAULT_QUEUE_ID } from 'app/store/nanostores/queueId';
|
import { $queueId, DEFAULT_QUEUE_ID } from 'app/store/nanostores/queueId';
|
||||||
import { $store } from 'app/store/nanostores/store';
|
import { $store } from 'app/store/nanostores/store';
|
||||||
import { $toastMap } from 'app/store/nanostores/toastMap';
|
import { $toastMap } from 'app/store/nanostores/toastMap';
|
||||||
|
import { $videoUpsellComponent } from 'app/store/nanostores/videoUpsellComponent';
|
||||||
import { $whatsNew } from 'app/store/nanostores/whatsNew';
|
import { $whatsNew } from 'app/store/nanostores/whatsNew';
|
||||||
import { createStore } from 'app/store/store';
|
import { createStore } from 'app/store/store';
|
||||||
import type { PartialAppConfig } from 'app/types/invokeai';
|
|
||||||
import Loading from 'common/components/Loading/Loading';
|
import Loading from 'common/components/Loading/Loading';
|
||||||
import type { WorkflowSortOption, WorkflowTagCategory } from 'features/nodes/store/workflowLibrarySlice';
|
|
||||||
import {
|
import {
|
||||||
$workflowLibraryCategoriesOptions,
|
$workflowLibraryCategoriesOptions,
|
||||||
$workflowLibrarySortOptions,
|
$workflowLibrarySortOptions,
|
||||||
@@ -33,47 +30,13 @@ import {
|
|||||||
DEFAULT_WORKFLOW_LIBRARY_SORT_OPTIONS,
|
DEFAULT_WORKFLOW_LIBRARY_SORT_OPTIONS,
|
||||||
DEFAULT_WORKFLOW_LIBRARY_TAG_CATEGORIES,
|
DEFAULT_WORKFLOW_LIBRARY_TAG_CATEGORIES,
|
||||||
} from 'features/nodes/store/workflowLibrarySlice';
|
} from 'features/nodes/store/workflowLibrarySlice';
|
||||||
import type { WorkflowCategory } from 'features/nodes/types/workflow';
|
|
||||||
import type { ToastConfig } from 'features/toast/toast';
|
|
||||||
import type { PropsWithChildren, ReactNode } from 'react';
|
|
||||||
import React, { lazy, memo, useEffect, useLayoutEffect, useState } from 'react';
|
import React, { lazy, memo, useEffect, useLayoutEffect, useState } from 'react';
|
||||||
import { Provider } from 'react-redux';
|
import { Provider } from 'react-redux';
|
||||||
import { addMiddleware, resetMiddlewares } from 'redux-dynamic-middlewares';
|
import { addMiddleware, resetMiddlewares } from 'redux-dynamic-middlewares';
|
||||||
import { $socketOptions } from 'services/events/stores';
|
import { $socketOptions } from 'services/events/stores';
|
||||||
import type { ManagerOptions, SocketOptions } from 'socket.io-client';
|
|
||||||
|
|
||||||
const App = lazy(() => import('./App'));
|
const App = lazy(() => import('./App'));
|
||||||
|
|
||||||
interface Props extends PropsWithChildren {
|
|
||||||
apiUrl?: string;
|
|
||||||
openAPISchemaUrl?: string;
|
|
||||||
token?: string;
|
|
||||||
config?: PartialAppConfig;
|
|
||||||
customNavComponent?: ReactNode;
|
|
||||||
accountSettingsLink?: string;
|
|
||||||
middleware?: Middleware[];
|
|
||||||
projectId?: string;
|
|
||||||
projectName?: string;
|
|
||||||
projectUrl?: string;
|
|
||||||
queueId?: string;
|
|
||||||
studioInitAction?: StudioInitAction;
|
|
||||||
customStarUi?: CustomStarUi;
|
|
||||||
socketOptions?: Partial<ManagerOptions & SocketOptions>;
|
|
||||||
isDebugging?: boolean;
|
|
||||||
logo?: ReactNode;
|
|
||||||
toastMap?: Record<string, ToastConfig>;
|
|
||||||
whatsNew?: ReactNode[];
|
|
||||||
workflowCategories?: WorkflowCategory[];
|
|
||||||
workflowTagCategories?: WorkflowTagCategory[];
|
|
||||||
workflowSortOptions?: WorkflowSortOption[];
|
|
||||||
loggingOverrides?: LoggingOverrides;
|
|
||||||
/**
|
|
||||||
* If provided, overrides in-app navigation to the model manager
|
|
||||||
*/
|
|
||||||
onClickGoToModelManager?: () => void;
|
|
||||||
storagePersistDebounce?: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
const InvokeAIUI = ({
|
const InvokeAIUI = ({
|
||||||
apiUrl,
|
apiUrl,
|
||||||
openAPISchemaUrl,
|
openAPISchemaUrl,
|
||||||
@@ -92,6 +55,8 @@ const InvokeAIUI = ({
|
|||||||
isDebugging = false,
|
isDebugging = false,
|
||||||
logo,
|
logo,
|
||||||
toastMap,
|
toastMap,
|
||||||
|
accountTypeText,
|
||||||
|
videoUpsellComponent,
|
||||||
workflowCategories,
|
workflowCategories,
|
||||||
workflowTagCategories,
|
workflowTagCategories,
|
||||||
workflowSortOptions,
|
workflowSortOptions,
|
||||||
@@ -99,7 +64,7 @@ const InvokeAIUI = ({
|
|||||||
onClickGoToModelManager,
|
onClickGoToModelManager,
|
||||||
whatsNew,
|
whatsNew,
|
||||||
storagePersistDebounce = 300,
|
storagePersistDebounce = 300,
|
||||||
}: Props) => {
|
}: InvokeAIUIProps) => {
|
||||||
const [store, setStore] = useState<ReturnType<typeof createStore> | undefined>(undefined);
|
const [store, setStore] = useState<ReturnType<typeof createStore> | undefined>(undefined);
|
||||||
const [didRehydrate, setDidRehydrate] = useState(false);
|
const [didRehydrate, setDidRehydrate] = useState(false);
|
||||||
|
|
||||||
@@ -180,6 +145,26 @@ const InvokeAIUI = ({
|
|||||||
};
|
};
|
||||||
}, [customStarUi]);
|
}, [customStarUi]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (accountTypeText) {
|
||||||
|
$accountTypeText.set(accountTypeText);
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
$accountTypeText.set('');
|
||||||
|
};
|
||||||
|
}, [accountTypeText]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (videoUpsellComponent) {
|
||||||
|
$videoUpsellComponent.set(videoUpsellComponent);
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
$videoUpsellComponent.set(undefined);
|
||||||
|
};
|
||||||
|
}, [videoUpsellComponent]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (customNavComponent) {
|
if (customNavComponent) {
|
||||||
$customNavComponent.set(customNavComponent);
|
$customNavComponent.set(customNavComponent);
|
||||||
|
|||||||
43
invokeai/frontend/web/src/app/components/types.ts
Normal file
43
invokeai/frontend/web/src/app/components/types.ts
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
import type { Middleware } from '@reduxjs/toolkit';
|
||||||
|
import type { StudioInitAction } from 'app/hooks/useStudioInitAction';
|
||||||
|
import type { LoggingOverrides } from 'app/logging/logger';
|
||||||
|
import type { CustomStarUi } from 'app/store/nanostores/customStarUI';
|
||||||
|
import type { PartialAppConfig } from 'app/types/invokeai';
|
||||||
|
import type { SocketOptions } from 'dgram';
|
||||||
|
import type { WorkflowSortOption, WorkflowTagCategory } from 'features/nodes/store/workflowLibrarySlice';
|
||||||
|
import type { WorkflowCategory } from 'features/nodes/types/workflow';
|
||||||
|
import type { ToastConfig } from 'features/toast/toast';
|
||||||
|
import type { PropsWithChildren, ReactNode } from 'react';
|
||||||
|
import type { ManagerOptions } from 'socket.io-client';
|
||||||
|
|
||||||
|
export interface InvokeAIUIProps extends PropsWithChildren {
|
||||||
|
apiUrl?: string;
|
||||||
|
openAPISchemaUrl?: string;
|
||||||
|
token?: string;
|
||||||
|
config?: PartialAppConfig;
|
||||||
|
customNavComponent?: ReactNode;
|
||||||
|
accountSettingsLink?: string;
|
||||||
|
middleware?: Middleware[];
|
||||||
|
projectId?: string;
|
||||||
|
projectName?: string;
|
||||||
|
projectUrl?: string;
|
||||||
|
queueId?: string;
|
||||||
|
studioInitAction?: StudioInitAction;
|
||||||
|
customStarUi?: CustomStarUi;
|
||||||
|
socketOptions?: Partial<ManagerOptions & SocketOptions>;
|
||||||
|
isDebugging?: boolean;
|
||||||
|
logo?: ReactNode;
|
||||||
|
toastMap?: Record<string, ToastConfig>;
|
||||||
|
accountTypeText?: string;
|
||||||
|
videoUpsellComponent?: ReactNode;
|
||||||
|
whatsNew?: ReactNode[];
|
||||||
|
workflowCategories?: WorkflowCategory[];
|
||||||
|
workflowTagCategories?: WorkflowTagCategory[];
|
||||||
|
workflowSortOptions?: WorkflowSortOption[];
|
||||||
|
loggingOverrides?: LoggingOverrides;
|
||||||
|
/**
|
||||||
|
* If provided, overrides in-app navigation to the model manager
|
||||||
|
*/
|
||||||
|
onClickGoToModelManager?: () => void;
|
||||||
|
storagePersistDebounce?: number;
|
||||||
|
}
|
||||||
@@ -4,7 +4,6 @@ import { useAssertSingleton } from 'common/hooks/useAssertSingleton';
|
|||||||
import { withResultAsync } from 'common/util/result';
|
import { withResultAsync } from 'common/util/result';
|
||||||
import { canvasReset } from 'features/controlLayers/store/actions';
|
import { canvasReset } from 'features/controlLayers/store/actions';
|
||||||
import { rasterLayerAdded } from 'features/controlLayers/store/canvasSlice';
|
import { rasterLayerAdded } from 'features/controlLayers/store/canvasSlice';
|
||||||
import { paramsReset } from 'features/controlLayers/store/paramsSlice';
|
|
||||||
import type { CanvasRasterLayerState } from 'features/controlLayers/store/types';
|
import type { CanvasRasterLayerState } from 'features/controlLayers/store/types';
|
||||||
import { imageDTOToImageObject } from 'features/controlLayers/store/util';
|
import { imageDTOToImageObject } from 'features/controlLayers/store/util';
|
||||||
import { sentImageToCanvas } from 'features/gallery/store/actions';
|
import { sentImageToCanvas } from 'features/gallery/store/actions';
|
||||||
@@ -42,6 +41,7 @@ type StudioDestinationAction = _StudioInitAction<
|
|||||||
| 'canvas'
|
| 'canvas'
|
||||||
| 'workflows'
|
| 'workflows'
|
||||||
| 'upscaling'
|
| 'upscaling'
|
||||||
|
| 'video'
|
||||||
| 'viewAllWorkflows'
|
| 'viewAllWorkflows'
|
||||||
| 'viewAllWorkflowsRecommended'
|
| 'viewAllWorkflowsRecommended'
|
||||||
| 'viewAllStylePresets';
|
| 'viewAllStylePresets';
|
||||||
@@ -118,7 +118,7 @@ export const useStudioInitAction = (action?: StudioInitAction) => {
|
|||||||
const metadata = getImageMetadataResult.value;
|
const metadata = getImageMetadataResult.value;
|
||||||
store.dispatch(canvasReset());
|
store.dispatch(canvasReset());
|
||||||
// This shows a toast
|
// This shows a toast
|
||||||
await MetadataUtils.recallAll(metadata, store);
|
await MetadataUtils.recallAllImageMetadata(metadata, store);
|
||||||
},
|
},
|
||||||
[store, t]
|
[store, t]
|
||||||
);
|
);
|
||||||
@@ -163,7 +163,6 @@ export const useStudioInitAction = (action?: StudioInitAction) => {
|
|||||||
case 'generation':
|
case 'generation':
|
||||||
// Go to the generate tab, open the launchpad
|
// Go to the generate tab, open the launchpad
|
||||||
await navigationApi.focusPanel('generate', LAUNCHPAD_PANEL_ID);
|
await navigationApi.focusPanel('generate', LAUNCHPAD_PANEL_ID);
|
||||||
store.dispatch(paramsReset());
|
|
||||||
break;
|
break;
|
||||||
case 'canvas':
|
case 'canvas':
|
||||||
// Go to the canvas tab, open the launchpad
|
// Go to the canvas tab, open the launchpad
|
||||||
@@ -177,6 +176,10 @@ export const useStudioInitAction = (action?: StudioInitAction) => {
|
|||||||
// Go to the upscaling tab
|
// Go to the upscaling tab
|
||||||
navigationApi.switchToTab('upscaling');
|
navigationApi.switchToTab('upscaling');
|
||||||
break;
|
break;
|
||||||
|
case 'video':
|
||||||
|
// Go to the video tab
|
||||||
|
await navigationApi.focusPanel('video', LAUNCHPAD_PANEL_ID);
|
||||||
|
break;
|
||||||
case 'viewAllWorkflows':
|
case 'viewAllWorkflows':
|
||||||
// Go to the workflows tab and open the workflow library modal
|
// Go to the workflows tab and open the workflow library modal
|
||||||
navigationApi.switchToTab('workflows');
|
navigationApi.switchToTab('workflows');
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ export const zLogNamespace = z.enum([
|
|||||||
'system',
|
'system',
|
||||||
'queue',
|
'queue',
|
||||||
'workflows',
|
'workflows',
|
||||||
|
'video',
|
||||||
]);
|
]);
|
||||||
export type LogNamespace = z.infer<typeof zLogNamespace>;
|
export type LogNamespace = z.infer<typeof zLogNamespace>;
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { createAction } from '@reduxjs/toolkit';
|
import { createAction } from '@reduxjs/toolkit';
|
||||||
import type { AppStartListening } from 'app/store/store';
|
import type { AppStartListening } from 'app/store/store';
|
||||||
import { selectLastSelectedImage } from 'features/gallery/store/gallerySelectors';
|
import { selectLastSelectedItem } from 'features/gallery/store/gallerySelectors';
|
||||||
import { imageSelected } from 'features/gallery/store/gallerySlice';
|
import { itemSelected } from 'features/gallery/store/gallerySlice';
|
||||||
import { imagesApi } from 'services/api/endpoints/images';
|
import { imagesApi } from 'services/api/endpoints/images';
|
||||||
|
|
||||||
export const appStarted = createAction('app/appStarted');
|
export const appStarted = createAction('app/appStarted');
|
||||||
@@ -18,11 +18,13 @@ export const addAppStartedListener = (startAppListening: AppStartListening) => {
|
|||||||
const firstImageLoad = await take(imagesApi.endpoints.getImageNames.matchFulfilled);
|
const firstImageLoad = await take(imagesApi.endpoints.getImageNames.matchFulfilled);
|
||||||
if (firstImageLoad !== null) {
|
if (firstImageLoad !== null) {
|
||||||
const [{ payload }] = firstImageLoad;
|
const [{ payload }] = firstImageLoad;
|
||||||
const selectedImage = selectLastSelectedImage(getState());
|
const selectedImage = selectLastSelectedItem(getState());
|
||||||
if (selectedImage) {
|
if (selectedImage) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
dispatch(imageSelected(payload.image_names.at(0) ?? null));
|
if (payload.image_names[0]) {
|
||||||
|
dispatch(itemSelected({ type: 'image', id: payload.image_names[0] }));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,8 +1,14 @@
|
|||||||
import { isAnyOf } from '@reduxjs/toolkit';
|
import { isAnyOf } from '@reduxjs/toolkit';
|
||||||
import type { AppStartListening } from 'app/store/store';
|
import type { AppStartListening } from 'app/store/store';
|
||||||
import { selectGetImageNamesQueryArgs, selectSelectedBoardId } from 'features/gallery/store/gallerySelectors';
|
import {
|
||||||
import { boardIdSelected, galleryViewChanged, imageSelected } from 'features/gallery/store/gallerySlice';
|
selectGalleryView,
|
||||||
|
selectGetImageNamesQueryArgs,
|
||||||
|
selectGetVideoIdsQueryArgs,
|
||||||
|
selectSelectedBoardId,
|
||||||
|
} from 'features/gallery/store/gallerySelectors';
|
||||||
|
import { boardIdSelected, galleryViewChanged, itemSelected } from 'features/gallery/store/gallerySlice';
|
||||||
import { imagesApi } from 'services/api/endpoints/images';
|
import { imagesApi } from 'services/api/endpoints/images';
|
||||||
|
import { videosApi } from 'services/api/endpoints/videos';
|
||||||
|
|
||||||
export const addBoardIdSelectedListener = (startAppListening: AppStartListening) => {
|
export const addBoardIdSelectedListener = (startAppListening: AppStartListening) => {
|
||||||
startAppListening({
|
startAppListening({
|
||||||
@@ -11,35 +17,65 @@ export const addBoardIdSelectedListener = (startAppListening: AppStartListening)
|
|||||||
// Cancel any in-progress instances of this listener, we don't want to select an image from a previous board
|
// Cancel any in-progress instances of this listener, we don't want to select an image from a previous board
|
||||||
cancelActiveListeners();
|
cancelActiveListeners();
|
||||||
|
|
||||||
if (boardIdSelected.match(action) && action.payload.selectedImageName) {
|
if (boardIdSelected.match(action) && action.payload.select) {
|
||||||
// This action already has a selected image name, we trust it is valid
|
// This action already has a resource selection - skip the below auto-selection logic
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const state = getState();
|
const state = getState();
|
||||||
|
|
||||||
const board_id = selectSelectedBoardId(state);
|
const board_id = selectSelectedBoardId(state);
|
||||||
|
const view = selectGalleryView(state);
|
||||||
|
|
||||||
const queryArgs = { ...selectGetImageNamesQueryArgs(state), board_id };
|
if (view === 'images' || view === 'assets') {
|
||||||
|
const queryArgs = { ...selectGetImageNamesQueryArgs(state), board_id };
|
||||||
|
// wait until the board has some images - maybe it already has some from a previous fetch
|
||||||
|
// must use getState() to ensure we do not have stale state
|
||||||
|
const isSuccess = await condition(
|
||||||
|
() => imagesApi.endpoints.getImageNames.select(queryArgs)(getState()).isSuccess,
|
||||||
|
5000
|
||||||
|
);
|
||||||
|
|
||||||
// wait until the board has some images - maybe it already has some from a previous fetch
|
if (!isSuccess) {
|
||||||
// must use getState() to ensure we do not have stale state
|
dispatch(itemSelected(null));
|
||||||
const isSuccess = await condition(
|
return;
|
||||||
() => imagesApi.endpoints.getImageNames.select(queryArgs)(getState()).isSuccess,
|
}
|
||||||
5000
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!isSuccess) {
|
// the board was just changed - we can select the first image
|
||||||
dispatch(imageSelected(null));
|
const imageNames = imagesApi.endpoints.getImageNames.select(queryArgs)(getState()).data?.image_names;
|
||||||
return;
|
|
||||||
|
const imageToSelect = imageNames && imageNames.length > 0 ? imageNames[0] : null;
|
||||||
|
|
||||||
|
if (imageToSelect) {
|
||||||
|
dispatch(itemSelected({ type: 'image', id: imageToSelect }));
|
||||||
|
} else {
|
||||||
|
dispatch(itemSelected(null));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const queryArgs = { ...selectGetVideoIdsQueryArgs(state), board_id };
|
||||||
|
// wait until the board has some images - maybe it already has some from a previous fetch
|
||||||
|
// must use getState() to ensure we do not have stale state
|
||||||
|
const isSuccess = await condition(
|
||||||
|
() => videosApi.endpoints.getVideoIds.select(queryArgs)(getState()).isSuccess,
|
||||||
|
5000
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!isSuccess) {
|
||||||
|
dispatch(itemSelected(null));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// the board was just changed - we can select the first image
|
||||||
|
const videoIds = videosApi.endpoints.getVideoIds.select(queryArgs)(getState()).data?.video_ids;
|
||||||
|
|
||||||
|
const videoToSelect = videoIds && videoIds.length > 0 ? videoIds[0] : null;
|
||||||
|
|
||||||
|
if (videoToSelect) {
|
||||||
|
dispatch(itemSelected({ type: 'video', id: videoToSelect }));
|
||||||
|
} else {
|
||||||
|
dispatch(itemSelected(null));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// the board was just changed - we can select the first image
|
|
||||||
const imageNames = imagesApi.endpoints.getImageNames.select(queryArgs)(getState()).data?.image_names;
|
|
||||||
|
|
||||||
const imageToSelect = imageNames?.at(0) ?? null;
|
|
||||||
|
|
||||||
dispatch(imageSelected(imageToSelect));
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -12,13 +12,21 @@ import {
|
|||||||
} from 'features/controlLayers/store/paramsSlice';
|
} from 'features/controlLayers/store/paramsSlice';
|
||||||
import { refImageModelChanged, selectRefImagesSlice } from 'features/controlLayers/store/refImagesSlice';
|
import { refImageModelChanged, selectRefImagesSlice } from 'features/controlLayers/store/refImagesSlice';
|
||||||
import { selectCanvasSlice } from 'features/controlLayers/store/selectors';
|
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 { modelSelected } from 'features/parameters/store/actions';
|
||||||
import {
|
import {
|
||||||
postProcessingModelChanged,
|
postProcessingModelChanged,
|
||||||
tileControlnetModelChanged,
|
tileControlnetModelChanged,
|
||||||
upscaleModelChanged,
|
upscaleModelChanged,
|
||||||
} from 'features/parameters/store/upscaleSlice';
|
} from 'features/parameters/store/upscaleSlice';
|
||||||
|
import { videoModelChanged } from 'features/parameters/store/videoSlice';
|
||||||
import {
|
import {
|
||||||
zParameterCLIPEmbedModel,
|
zParameterCLIPEmbedModel,
|
||||||
zParameterSpandrelImageToImageModel,
|
zParameterSpandrelImageToImageModel,
|
||||||
@@ -41,6 +49,7 @@ import {
|
|||||||
isRefinerMainModelModelConfig,
|
isRefinerMainModelModelConfig,
|
||||||
isSpandrelImageToImageModelConfig,
|
isSpandrelImageToImageModelConfig,
|
||||||
isT5EncoderModelConfig,
|
isT5EncoderModelConfig,
|
||||||
|
isVideoModelConfig,
|
||||||
} from 'services/api/types';
|
} from 'services/api/types';
|
||||||
import type { JsonObject } from 'type-fest';
|
import type { JsonObject } from 'type-fest';
|
||||||
|
|
||||||
@@ -81,6 +90,7 @@ export const addModelsLoadedListener = (startAppListening: AppStartListening) =>
|
|||||||
handleCLIPEmbedModels(models, state, dispatch, log);
|
handleCLIPEmbedModels(models, state, dispatch, log);
|
||||||
handleFLUXVAEModels(models, state, dispatch, log);
|
handleFLUXVAEModels(models, state, dispatch, log);
|
||||||
handleFLUXReduxModels(models, state, dispatch, log);
|
handleFLUXReduxModels(models, state, dispatch, log);
|
||||||
|
handleVideoModels(models, state, dispatch, log);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
@@ -193,6 +203,22 @@ const handleLoRAModels: ModelHandler = (models, state, dispatch, log) => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleVideoModels: ModelHandler = (models, state, dispatch, log) => {
|
||||||
|
const videoModels = models.filter(isVideoModelConfig);
|
||||||
|
const selectedVideoModel = state.video.videoModel;
|
||||||
|
|
||||||
|
if (selectedVideoModel && videoModels.some((m) => m.key === selectedVideoModel.key)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const firstModel = videoModels[0] || null;
|
||||||
|
if (firstModel) {
|
||||||
|
log.debug({ firstModel }, 'No video model selected, selecting first available video model');
|
||||||
|
dispatch(videoModelChanged({ videoModel: zModelIdentifierField.parse(firstModel) }));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleControlAdapterModels: ModelHandler = (models, state, dispatch, log) => {
|
const handleControlAdapterModels: ModelHandler = (models, state, dispatch, log) => {
|
||||||
const caModels = models.filter(isControlLayerModelConfig);
|
const caModels = models.filter(isControlLayerModelConfig);
|
||||||
selectCanvasSlice(state).controlLayers.entities.forEach((entity) => {
|
selectCanvasSlice(state).controlLayers.entities.forEach((entity) => {
|
||||||
@@ -232,7 +258,7 @@ const handleIPAdapterModels: ModelHandler = (models, state, dispatch, log) => {
|
|||||||
|
|
||||||
selectCanvasSlice(state).regionalGuidance.entities.forEach((entity) => {
|
selectCanvasSlice(state).regionalGuidance.entities.forEach((entity) => {
|
||||||
entity.referenceImages.forEach(({ id: referenceImageId, config }) => {
|
entity.referenceImages.forEach(({ id: referenceImageId, config }) => {
|
||||||
if (!isIPAdapterConfig(config)) {
|
if (!isRegionalGuidanceIPAdapterConfig(config)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -275,7 +301,7 @@ const handleFLUXReduxModels: ModelHandler = (models, state, dispatch, log) => {
|
|||||||
|
|
||||||
selectCanvasSlice(state).regionalGuidance.entities.forEach((entity) => {
|
selectCanvasSlice(state).regionalGuidance.entities.forEach((entity) => {
|
||||||
entity.referenceImages.forEach(({ id: referenceImageId, config }) => {
|
entity.referenceImages.forEach(({ id: referenceImageId, config }) => {
|
||||||
if (!isFLUXReduxConfig(config)) {
|
if (!isRegionalGuidanceFLUXReduxConfig(config)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,3 @@
|
|||||||
|
import { atom } from 'nanostores';
|
||||||
|
|
||||||
|
export const $accountTypeText = atom<string>('');
|
||||||
@@ -1,6 +1,11 @@
|
|||||||
import { atom } from 'nanostores';
|
import { atom, computed } from 'nanostores';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The user's auth token.
|
* The user's auth token.
|
||||||
*/
|
*/
|
||||||
export const $authToken = atom<string | undefined>();
|
export const $authToken = atom<string | undefined>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The crossOrigin value to use for all image loading. Depends on whether the user is authenticated.
|
||||||
|
*/
|
||||||
|
export const $crossOrigin = computed($authToken, (token) => (token ? 'use-credentials' : 'anonymous'));
|
||||||
|
|||||||
@@ -0,0 +1,4 @@
|
|||||||
|
import { atom } from 'nanostores';
|
||||||
|
import type { ReactNode } from 'react';
|
||||||
|
|
||||||
|
export const $videoUpsellComponent = atom<ReactNode | undefined>(undefined);
|
||||||
@@ -18,7 +18,8 @@ import { addModelsLoadedListener } from 'app/store/middleware/listenerMiddleware
|
|||||||
import { addSetDefaultSettingsListener } from 'app/store/middleware/listenerMiddleware/listeners/setDefaultSettings';
|
import { addSetDefaultSettingsListener } from 'app/store/middleware/listenerMiddleware/listeners/setDefaultSettings';
|
||||||
import { addSocketConnectedEventListener } from 'app/store/middleware/listenerMiddleware/listeners/socketConnected';
|
import { addSocketConnectedEventListener } from 'app/store/middleware/listenerMiddleware/listeners/socketConnected';
|
||||||
import { deepClone } from 'common/util/deepClone';
|
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 { changeBoardModalSliceConfig } from 'features/changeBoardModal/store/slice';
|
||||||
import { canvasSettingsSliceConfig } from 'features/controlLayers/store/canvasSettingsSlice';
|
import { canvasSettingsSliceConfig } from 'features/controlLayers/store/canvasSettingsSlice';
|
||||||
import { canvasSliceConfig } from 'features/controlLayers/store/canvasSlice';
|
import { canvasSliceConfig } from 'features/controlLayers/store/canvasSlice';
|
||||||
@@ -33,6 +34,7 @@ import { nodesSliceConfig } from 'features/nodes/store/nodesSlice';
|
|||||||
import { workflowLibrarySliceConfig } from 'features/nodes/store/workflowLibrarySlice';
|
import { workflowLibrarySliceConfig } from 'features/nodes/store/workflowLibrarySlice';
|
||||||
import { workflowSettingsSliceConfig } from 'features/nodes/store/workflowSettingsSlice';
|
import { workflowSettingsSliceConfig } from 'features/nodes/store/workflowSettingsSlice';
|
||||||
import { upscaleSliceConfig } from 'features/parameters/store/upscaleSlice';
|
import { upscaleSliceConfig } from 'features/parameters/store/upscaleSlice';
|
||||||
|
import { videoSliceConfig } from 'features/parameters/store/videoSlice';
|
||||||
import { queueSliceConfig } from 'features/queue/store/queueSlice';
|
import { queueSliceConfig } from 'features/queue/store/queueSlice';
|
||||||
import { stylePresetSliceConfig } from 'features/stylePresets/store/stylePresetSlice';
|
import { stylePresetSliceConfig } from 'features/stylePresets/store/stylePresetSlice';
|
||||||
import { configSliceConfig } from 'features/system/store/configSlice';
|
import { configSliceConfig } from 'features/system/store/configSlice';
|
||||||
@@ -78,6 +80,7 @@ const SLICE_CONFIGS = {
|
|||||||
[systemSliceConfig.slice.reducerPath]: systemSliceConfig,
|
[systemSliceConfig.slice.reducerPath]: systemSliceConfig,
|
||||||
[uiSliceConfig.slice.reducerPath]: uiSliceConfig,
|
[uiSliceConfig.slice.reducerPath]: uiSliceConfig,
|
||||||
[upscaleSliceConfig.slice.reducerPath]: upscaleSliceConfig,
|
[upscaleSliceConfig.slice.reducerPath]: upscaleSliceConfig,
|
||||||
|
[videoSliceConfig.slice.reducerPath]: videoSliceConfig,
|
||||||
[workflowLibrarySliceConfig.slice.reducerPath]: workflowLibrarySliceConfig,
|
[workflowLibrarySliceConfig.slice.reducerPath]: workflowLibrarySliceConfig,
|
||||||
[workflowSettingsSliceConfig.slice.reducerPath]: workflowSettingsSliceConfig,
|
[workflowSettingsSliceConfig.slice.reducerPath]: workflowSettingsSliceConfig,
|
||||||
};
|
};
|
||||||
@@ -111,6 +114,7 @@ const ALL_REDUCERS = {
|
|||||||
[systemSliceConfig.slice.reducerPath]: systemSliceConfig.slice.reducer,
|
[systemSliceConfig.slice.reducerPath]: systemSliceConfig.slice.reducer,
|
||||||
[uiSliceConfig.slice.reducerPath]: uiSliceConfig.slice.reducer,
|
[uiSliceConfig.slice.reducerPath]: uiSliceConfig.slice.reducer,
|
||||||
[upscaleSliceConfig.slice.reducerPath]: upscaleSliceConfig.slice.reducer,
|
[upscaleSliceConfig.slice.reducerPath]: upscaleSliceConfig.slice.reducer,
|
||||||
|
[videoSliceConfig.slice.reducerPath]: videoSliceConfig.slice.reducer,
|
||||||
[workflowLibrarySliceConfig.slice.reducerPath]: workflowLibrarySliceConfig.slice.reducer,
|
[workflowLibrarySliceConfig.slice.reducerPath]: workflowLibrarySliceConfig.slice.reducer,
|
||||||
[workflowSettingsSliceConfig.slice.reducerPath]: workflowSettingsSliceConfig.slice.reducer,
|
[workflowSettingsSliceConfig.slice.reducerPath]: workflowSettingsSliceConfig.slice.reducer,
|
||||||
};
|
};
|
||||||
@@ -130,16 +134,14 @@ const unserialize: UnserializeFunction = (data, key) => {
|
|||||||
const initialState = getInitialState();
|
const initialState = getInitialState();
|
||||||
const parsed = JSON.parse(data);
|
const parsed = JSON.parse(data);
|
||||||
|
|
||||||
// strip out old keys
|
// We need to inject non-persisted values from initial state into the rehydrated state. These values always are
|
||||||
const stripped = pick(deepClone(parsed), keys(initialState));
|
// 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.
|
||||||
* Merge in initial state as default values, covering any missing keys. You might be tempted to use _.defaultsDeep,
|
const nonPersistedSubsetOfState = pick(initialState, persistConfig.persistDenylist ?? []);
|
||||||
* but that merges arrays by index and partial objects by key. Using an identity function as the customizer results
|
const stateToMigrate = merge(deepClone(parsed), nonPersistedSubsetOfState);
|
||||||
* in behaviour like defaultsDeep, but doesn't overwrite any values that are not undefined in the migrated state.
|
|
||||||
*/
|
// Run migrations to bring old state up to date with the current version.
|
||||||
const unPersistDenylisted = mergeWith(stripped, initialState, (objVal) => objVal);
|
const migrated = persistConfig.migrate(stateToMigrate);
|
||||||
// run (additive) migrations
|
|
||||||
const migrated = persistConfig.migrate(unPersistDenylisted);
|
|
||||||
|
|
||||||
log.debug(
|
log.debug(
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -80,6 +80,7 @@ export const zAppConfig = z.object({
|
|||||||
allowClientSideUpload: z.boolean(),
|
allowClientSideUpload: z.boolean(),
|
||||||
allowPublishWorkflows: z.boolean(),
|
allowPublishWorkflows: z.boolean(),
|
||||||
allowPromptExpansion: z.boolean(),
|
allowPromptExpansion: z.boolean(),
|
||||||
|
allowVideo: z.boolean(),
|
||||||
disabledTabs: z.array(zTabName),
|
disabledTabs: z.array(zTabName),
|
||||||
disabledFeatures: z.array(zAppFeature),
|
disabledFeatures: z.array(zAppFeature),
|
||||||
disabledSDFeatures: z.array(zSDFeature),
|
disabledSDFeatures: z.array(zSDFeature),
|
||||||
@@ -140,8 +141,9 @@ export const getDefaultAppConfig = (): AppConfig => ({
|
|||||||
allowClientSideUpload: false,
|
allowClientSideUpload: false,
|
||||||
allowPublishWorkflows: false,
|
allowPublishWorkflows: false,
|
||||||
allowPromptExpansion: false,
|
allowPromptExpansion: false,
|
||||||
|
allowVideo: false, // used to determine if video is enabled vs upsell
|
||||||
shouldShowCredits: false,
|
shouldShowCredits: false,
|
||||||
disabledTabs: [],
|
disabledTabs: ['video'], // used to determine if video functionality is visible
|
||||||
disabledFeatures: ['lightbox', 'faceRestore', 'batches'] satisfies AppFeature[],
|
disabledFeatures: ['lightbox', 'faceRestore', 'batches'] satisfies AppFeature[],
|
||||||
disabledSDFeatures: ['variation', 'symmetry', 'hires', 'perlinNoise', 'noiseThreshold'] satisfies SDFeature[],
|
disabledSDFeatures: ['variation', 'symmetry', 'hires', 'perlinNoise', 'noiseThreshold'] satisfies SDFeature[],
|
||||||
sd: {
|
sd: {
|
||||||
|
|||||||
@@ -37,6 +37,7 @@ const REGION_NAMES = [
|
|||||||
'workflows',
|
'workflows',
|
||||||
'progress',
|
'progress',
|
||||||
'settings',
|
'settings',
|
||||||
|
'video',
|
||||||
] as const;
|
] as const;
|
||||||
/**
|
/**
|
||||||
* The names of the focus regions.
|
* The names of the focus regions.
|
||||||
|
|||||||
@@ -6,13 +6,13 @@ import { toast } from 'features/toast/toast';
|
|||||||
import { useCallback } from 'react';
|
import { useCallback } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
|
||||||
export const useDownloadImage = () => {
|
export const useDownloadItem = () => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const authToken = useStore($authToken);
|
const authToken = useStore($authToken);
|
||||||
|
|
||||||
const downloadImage = useCallback(
|
const downloadItem = useCallback(
|
||||||
async (image_url: string, image_name: string) => {
|
async (item_url: string, item_id: string) => {
|
||||||
try {
|
try {
|
||||||
const requestOpts = authToken
|
const requestOpts = authToken
|
||||||
? {
|
? {
|
||||||
@@ -21,7 +21,7 @@ export const useDownloadImage = () => {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
: {};
|
: {};
|
||||||
const blob = await fetch(image_url, requestOpts).then((resp) => resp.blob());
|
const blob = await fetch(item_url, requestOpts).then((resp) => resp.blob());
|
||||||
if (!blob) {
|
if (!blob) {
|
||||||
throw new Error('Unable to create Blob');
|
throw new Error('Unable to create Blob');
|
||||||
}
|
}
|
||||||
@@ -30,7 +30,7 @@ export const useDownloadImage = () => {
|
|||||||
const a = document.createElement('a');
|
const a = document.createElement('a');
|
||||||
a.style.display = 'none';
|
a.style.display = 'none';
|
||||||
a.href = url;
|
a.href = url;
|
||||||
a.download = image_name;
|
a.download = item_id;
|
||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
window.URL.revokeObjectURL(url);
|
window.URL.revokeObjectURL(url);
|
||||||
@@ -47,5 +47,5 @@ export const useDownloadImage = () => {
|
|||||||
[t, dispatch, authToken]
|
[t, dispatch, authToken]
|
||||||
);
|
);
|
||||||
|
|
||||||
return { downloadImage };
|
return { downloadItem };
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { useAppStore } from 'app/store/storeHooks';
|
import { useAppStore } from 'app/store/storeHooks';
|
||||||
import { useDeleteImageModalApi } from 'features/deleteImageModal/store/state';
|
import { useDeleteImageModalApi } from 'features/deleteImageModal/store/state';
|
||||||
|
import { useDeleteVideoModalApi } from 'features/deleteVideoModal/store/state';
|
||||||
import { selectSelection } from 'features/gallery/store/gallerySelectors';
|
import { selectSelection } from 'features/gallery/store/gallerySelectors';
|
||||||
import { useClearQueue } from 'features/queue/hooks/useClearQueue';
|
import { useClearQueue } from 'features/queue/hooks/useClearQueue';
|
||||||
import { useDeleteCurrentQueueItem } from 'features/queue/hooks/useDeleteCurrentQueueItem';
|
import { useDeleteCurrentQueueItem } from 'features/queue/hooks/useDeleteCurrentQueueItem';
|
||||||
@@ -12,6 +13,7 @@ import { getFocusedRegion } from './focus';
|
|||||||
|
|
||||||
export const useGlobalHotkeys = () => {
|
export const useGlobalHotkeys = () => {
|
||||||
const { dispatch, getState } = useAppStore();
|
const { dispatch, getState } = useAppStore();
|
||||||
|
const isVideoEnabled = useFeatureStatus('video');
|
||||||
const isModelManagerEnabled = useFeatureStatus('modelManager');
|
const isModelManagerEnabled = useFeatureStatus('modelManager');
|
||||||
const queue = useInvoke();
|
const queue = useInvoke();
|
||||||
|
|
||||||
@@ -92,6 +94,18 @@ export const useGlobalHotkeys = () => {
|
|||||||
dependencies: [dispatch],
|
dependencies: [dispatch],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
useRegisteredHotkeys({
|
||||||
|
id: 'selectVideoTab',
|
||||||
|
category: 'app',
|
||||||
|
callback: () => {
|
||||||
|
navigationApi.switchToTab('video');
|
||||||
|
},
|
||||||
|
options: {
|
||||||
|
enabled: isVideoEnabled,
|
||||||
|
},
|
||||||
|
dependencies: [dispatch],
|
||||||
|
});
|
||||||
|
|
||||||
useRegisteredHotkeys({
|
useRegisteredHotkeys({
|
||||||
id: 'selectWorkflowsTab',
|
id: 'selectWorkflowsTab',
|
||||||
category: 'app',
|
category: 'app',
|
||||||
@@ -123,6 +137,8 @@ export const useGlobalHotkeys = () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const deleteImageModalApi = useDeleteImageModalApi();
|
const deleteImageModalApi = useDeleteImageModalApi();
|
||||||
|
const deleteVideoModalApi = useDeleteVideoModalApi();
|
||||||
|
|
||||||
useRegisteredHotkeys({
|
useRegisteredHotkeys({
|
||||||
id: 'deleteSelection',
|
id: 'deleteSelection',
|
||||||
category: 'gallery',
|
category: 'gallery',
|
||||||
@@ -135,7 +151,13 @@ export const useGlobalHotkeys = () => {
|
|||||||
if (!selection.length) {
|
if (!selection.length) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
deleteImageModalApi.delete(selection);
|
if (selection.every(({ type }) => type === 'image')) {
|
||||||
|
deleteImageModalApi.delete(selection.map((s) => s.id));
|
||||||
|
} else if (selection.every(({ type }) => type === 'video')) {
|
||||||
|
deleteVideoModalApi.delete(selection.map((s) => s.id));
|
||||||
|
} else {
|
||||||
|
// no-op, we expect selections to always be only images or only video
|
||||||
|
}
|
||||||
},
|
},
|
||||||
dependencies: [getState, deleteImageModalApi],
|
dependencies: [getState, deleteImageModalApi],
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -13,12 +13,18 @@ import { memo, useCallback, useMemo, useState } from 'react';
|
|||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { useListAllBoardsQuery } from 'services/api/endpoints/boards';
|
import { useListAllBoardsQuery } from 'services/api/endpoints/boards';
|
||||||
import { useAddImagesToBoardMutation, useRemoveImagesFromBoardMutation } from 'services/api/endpoints/images';
|
import { useAddImagesToBoardMutation, useRemoveImagesFromBoardMutation } from 'services/api/endpoints/images';
|
||||||
|
import { useAddVideosToBoardMutation, useRemoveVideosFromBoardMutation } from 'services/api/endpoints/videos';
|
||||||
|
|
||||||
const selectImagesToChange = createSelector(
|
const selectImagesToChange = createSelector(
|
||||||
selectChangeBoardModalSlice,
|
selectChangeBoardModalSlice,
|
||||||
(changeBoardModal) => changeBoardModal.image_names
|
(changeBoardModal) => changeBoardModal.image_names
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const selectVideosToChange = createSelector(
|
||||||
|
selectChangeBoardModalSlice,
|
||||||
|
(changeBoardModal) => changeBoardModal.video_ids
|
||||||
|
);
|
||||||
|
|
||||||
const selectIsModalOpen = createSelector(
|
const selectIsModalOpen = createSelector(
|
||||||
selectChangeBoardModalSlice,
|
selectChangeBoardModalSlice,
|
||||||
(changeBoardModal) => changeBoardModal.isModalOpen
|
(changeBoardModal) => changeBoardModal.isModalOpen
|
||||||
@@ -32,8 +38,11 @@ const ChangeBoardModal = () => {
|
|||||||
const { data: boards, isFetching } = useListAllBoardsQuery({ include_archived: true });
|
const { data: boards, isFetching } = useListAllBoardsQuery({ include_archived: true });
|
||||||
const isModalOpen = useAppSelector(selectIsModalOpen);
|
const isModalOpen = useAppSelector(selectIsModalOpen);
|
||||||
const imagesToChange = useAppSelector(selectImagesToChange);
|
const imagesToChange = useAppSelector(selectImagesToChange);
|
||||||
|
const videosToChange = useAppSelector(selectVideosToChange);
|
||||||
const [addImagesToBoard] = useAddImagesToBoardMutation();
|
const [addImagesToBoard] = useAddImagesToBoardMutation();
|
||||||
const [removeImagesFromBoard] = useRemoveImagesFromBoardMutation();
|
const [removeImagesFromBoard] = useRemoveImagesFromBoardMutation();
|
||||||
|
const [addVideosToBoard] = useAddVideosToBoardMutation();
|
||||||
|
const [removeVideosFromBoard] = useRemoveVideosFromBoardMutation();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
const options = useMemo<ComboboxOption[]>(() => {
|
const options = useMemo<ComboboxOption[]>(() => {
|
||||||
@@ -57,20 +66,41 @@ const ChangeBoardModal = () => {
|
|||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
const handleChangeBoard = useCallback(() => {
|
const handleChangeBoard = useCallback(() => {
|
||||||
if (!imagesToChange.length || !selectedBoardId) {
|
if (!selectedBoardId || (imagesToChange.length === 0 && videosToChange.length === 0)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (selectedBoardId === 'none') {
|
if (imagesToChange.length) {
|
||||||
removeImagesFromBoard({ image_names: imagesToChange });
|
if (selectedBoardId === 'none') {
|
||||||
} else {
|
removeImagesFromBoard({ image_names: imagesToChange });
|
||||||
addImagesToBoard({
|
} else {
|
||||||
image_names: imagesToChange,
|
addImagesToBoard({
|
||||||
board_id: selectedBoardId,
|
image_names: imagesToChange,
|
||||||
});
|
board_id: selectedBoardId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (videosToChange.length) {
|
||||||
|
if (selectedBoardId === 'none') {
|
||||||
|
removeVideosFromBoard({ video_ids: videosToChange });
|
||||||
|
} else {
|
||||||
|
addVideosToBoard({
|
||||||
|
video_ids: videosToChange,
|
||||||
|
board_id: selectedBoardId,
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
dispatch(changeBoardReset());
|
dispatch(changeBoardReset());
|
||||||
}, [addImagesToBoard, dispatch, imagesToChange, removeImagesFromBoard, selectedBoardId]);
|
}, [
|
||||||
|
addImagesToBoard,
|
||||||
|
dispatch,
|
||||||
|
imagesToChange,
|
||||||
|
videosToChange,
|
||||||
|
removeImagesFromBoard,
|
||||||
|
selectedBoardId,
|
||||||
|
addVideosToBoard,
|
||||||
|
removeVideosFromBoard,
|
||||||
|
]);
|
||||||
|
|
||||||
const onChange = useCallback<ComboboxOnChange>((v) => {
|
const onChange = useCallback<ComboboxOnChange>((v) => {
|
||||||
if (!v) {
|
if (!v) {
|
||||||
@@ -91,9 +121,15 @@ const ChangeBoardModal = () => {
|
|||||||
>
|
>
|
||||||
<Flex flexDir="column" gap={4}>
|
<Flex flexDir="column" gap={4}>
|
||||||
<Text>
|
<Text>
|
||||||
{t('boards.movingImagesToBoard', {
|
{imagesToChange.length > 0 &&
|
||||||
count: imagesToChange.length,
|
t('boards.movingImagesToBoard', {
|
||||||
})}
|
count: imagesToChange.length,
|
||||||
|
})}
|
||||||
|
{videosToChange.length > 0 &&
|
||||||
|
t('boards.movingVideosToBoard', {
|
||||||
|
count: videosToChange.length,
|
||||||
|
})}
|
||||||
|
:
|
||||||
</Text>
|
</Text>
|
||||||
<FormControl isDisabled={isFetching}>
|
<FormControl isDisabled={isFetching}>
|
||||||
<Combobox
|
<Combobox
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import z from 'zod';
|
|||||||
const zChangeBoardModalState = z.object({
|
const zChangeBoardModalState = z.object({
|
||||||
isModalOpen: z.boolean().default(false),
|
isModalOpen: z.boolean().default(false),
|
||||||
image_names: z.array(z.string()).default(() => []),
|
image_names: z.array(z.string()).default(() => []),
|
||||||
|
video_ids: z.array(z.string()).default(() => []),
|
||||||
});
|
});
|
||||||
type ChangeBoardModalState = z.infer<typeof zChangeBoardModalState>;
|
type ChangeBoardModalState = z.infer<typeof zChangeBoardModalState>;
|
||||||
|
|
||||||
@@ -22,6 +23,9 @@ const slice = createSlice({
|
|||||||
imagesToChangeSelected: (state, action: PayloadAction<string[]>) => {
|
imagesToChangeSelected: (state, action: PayloadAction<string[]>) => {
|
||||||
state.image_names = action.payload;
|
state.image_names = action.payload;
|
||||||
},
|
},
|
||||||
|
videosToChangeSelected: (state, action: PayloadAction<string[]>) => {
|
||||||
|
state.video_ids = action.payload;
|
||||||
|
},
|
||||||
changeBoardReset: (state) => {
|
changeBoardReset: (state) => {
|
||||||
state.image_names = [];
|
state.image_names = [];
|
||||||
state.isModalOpen = false;
|
state.isModalOpen = false;
|
||||||
@@ -29,7 +33,7 @@ const slice = createSlice({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
export const { isModalOpenChanged, imagesToChangeSelected, changeBoardReset } = slice.actions;
|
export const { isModalOpenChanged, imagesToChangeSelected, videosToChangeSelected, changeBoardReset } = slice.actions;
|
||||||
|
|
||||||
export const selectChangeBoardModalSlice = (state: RootState) => state.changeBoardModal;
|
export const selectChangeBoardModalSlice = (state: RootState) => state.changeBoardModal;
|
||||||
|
|
||||||
|
|||||||
@@ -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 { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
import RgbColorPicker from 'common/components/ColorPicker/RgbColorPicker';
|
import RgbColorPicker from 'common/components/ColorPicker/RgbColorPicker';
|
||||||
import { rgbColorToString } from 'common/util/colorCodeTransformers';
|
import { rgbColorToString } from 'common/util/colorCodeTransformers';
|
||||||
@@ -62,14 +71,16 @@ export const EntityListSelectedEntityActionBarFill = memo(() => {
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Flex>
|
</Flex>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent>
|
<Portal>
|
||||||
<PopoverBody minH={64}>
|
<PopoverContent>
|
||||||
<Flex flexDir="column" gap={4}>
|
<PopoverBody minH={64}>
|
||||||
<RgbColorPicker color={fill.color} onChange={onChangeFillColor} withNumberInput withSwatches />
|
<Flex flexDir="column" gap={4}>
|
||||||
<MaskFillStyle style={fill.style} onChange={onChangeFillStyle} />
|
<RgbColorPicker color={fill.color} onChange={onChangeFillColor} withNumberInput withSwatches />
|
||||||
</Flex>
|
<MaskFillStyle style={fill.style} onChange={onChangeFillStyle} />
|
||||||
</PopoverBody>
|
</Flex>
|
||||||
</PopoverContent>
|
</PopoverBody>
|
||||||
|
</PopoverContent>
|
||||||
|
</Portal>
|
||||||
</Popover>
|
</Popover>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import {
|
|||||||
PopoverBody,
|
PopoverBody,
|
||||||
PopoverContent,
|
PopoverContent,
|
||||||
PopoverTrigger,
|
PopoverTrigger,
|
||||||
|
Portal,
|
||||||
} from '@invoke-ai/ui-library';
|
} from '@invoke-ai/ui-library';
|
||||||
import { createSelector } from '@reduxjs/toolkit';
|
import { createSelector } from '@reduxjs/toolkit';
|
||||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
@@ -165,22 +166,24 @@ export const EntityListSelectedEntityActionBarOpacity = memo(() => {
|
|||||||
</NumberInput>
|
</NumberInput>
|
||||||
</PopoverAnchor>
|
</PopoverAnchor>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
<Portal>
|
||||||
<PopoverArrow />
|
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||||
<PopoverBody>
|
<PopoverArrow />
|
||||||
<CompositeSlider
|
<PopoverBody>
|
||||||
min={0}
|
<CompositeSlider
|
||||||
max={100}
|
min={0}
|
||||||
value={localOpacity}
|
max={100}
|
||||||
onChange={onChangeSlider}
|
value={localOpacity}
|
||||||
defaultValue={sliderDefaultValue}
|
onChange={onChangeSlider}
|
||||||
marks={marks}
|
defaultValue={sliderDefaultValue}
|
||||||
formatValue={formatSliderValue}
|
marks={marks}
|
||||||
alwaysShowMarks
|
formatValue={formatSliderValue}
|
||||||
isDisabled={selectedEntityIdentifier === null}
|
alwaysShowMarks
|
||||||
/>
|
isDisabled={selectedEntityIdentifier === null}
|
||||||
</PopoverBody>
|
/>
|
||||||
</PopoverContent>
|
</PopoverBody>
|
||||||
|
</PopoverContent>
|
||||||
|
</Portal>
|
||||||
</Popover>
|
</Popover>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { CanvasEntityHeader } from 'features/controlLayers/components/common/Can
|
|||||||
import { CanvasEntityHeaderCommonActions } from 'features/controlLayers/components/common/CanvasEntityHeaderCommonActions';
|
import { CanvasEntityHeaderCommonActions } from 'features/controlLayers/components/common/CanvasEntityHeaderCommonActions';
|
||||||
import { CanvasEntityPreviewImage } from 'features/controlLayers/components/common/CanvasEntityPreviewImage';
|
import { CanvasEntityPreviewImage } from 'features/controlLayers/components/common/CanvasEntityPreviewImage';
|
||||||
import { CanvasEntityEditableTitle } from 'features/controlLayers/components/common/CanvasEntityTitleEdit';
|
import { CanvasEntityEditableTitle } from 'features/controlLayers/components/common/CanvasEntityTitleEdit';
|
||||||
|
import { RasterLayerAdjustmentsPanel } from 'features/controlLayers/components/RasterLayer/RasterLayerAdjustmentsPanel';
|
||||||
import { CanvasEntityStateGate } from 'features/controlLayers/contexts/CanvasEntityStateGate';
|
import { CanvasEntityStateGate } from 'features/controlLayers/contexts/CanvasEntityStateGate';
|
||||||
import { RasterLayerAdapterGate } from 'features/controlLayers/contexts/EntityAdapterContext';
|
import { RasterLayerAdapterGate } from 'features/controlLayers/contexts/EntityAdapterContext';
|
||||||
import { EntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
import { EntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||||
@@ -39,6 +40,7 @@ export const RasterLayer = memo(({ id }: Props) => {
|
|||||||
<Spacer />
|
<Spacer />
|
||||||
<CanvasEntityHeaderCommonActions />
|
<CanvasEntityHeaderCommonActions />
|
||||||
</CanvasEntityHeader>
|
</CanvasEntityHeader>
|
||||||
|
<RasterLayerAdjustmentsPanel />
|
||||||
<DndDropTarget
|
<DndDropTarget
|
||||||
dndTarget={replaceCanvasEntityObjectsWithImageDndTarget}
|
dndTarget={replaceCanvasEntityObjectsWithImageDndTarget}
|
||||||
dndTargetData={dndTargetData}
|
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 { CanvasEntityMenuItemsSave } from 'features/controlLayers/components/common/CanvasEntityMenuItemsSave';
|
||||||
import { CanvasEntityMenuItemsSelectObject } from 'features/controlLayers/components/common/CanvasEntityMenuItemsSelectObject';
|
import { CanvasEntityMenuItemsSelectObject } from 'features/controlLayers/components/common/CanvasEntityMenuItemsSelectObject';
|
||||||
import { CanvasEntityMenuItemsTransform } from 'features/controlLayers/components/common/CanvasEntityMenuItemsTransform';
|
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 { RasterLayerMenuItemsConvertToSubMenu } from 'features/controlLayers/components/RasterLayer/RasterLayerMenuItemsConvertToSubMenu';
|
||||||
import { RasterLayerMenuItemsCopyToSubMenu } from 'features/controlLayers/components/RasterLayer/RasterLayerMenuItemsCopyToSubMenu';
|
import { RasterLayerMenuItemsCopyToSubMenu } from 'features/controlLayers/components/RasterLayer/RasterLayerMenuItemsCopyToSubMenu';
|
||||||
import { memo } from 'react';
|
import { memo } from 'react';
|
||||||
@@ -21,10 +22,10 @@ export const RasterLayerMenuItems = memo(() => {
|
|||||||
<CanvasEntityMenuItemsDuplicate />
|
<CanvasEntityMenuItemsDuplicate />
|
||||||
<CanvasEntityMenuItemsDelete asIcon />
|
<CanvasEntityMenuItemsDelete asIcon />
|
||||||
</IconMenuItemGroup>
|
</IconMenuItemGroup>
|
||||||
<MenuDivider />
|
|
||||||
<CanvasEntityMenuItemsTransform />
|
<CanvasEntityMenuItemsTransform />
|
||||||
<CanvasEntityMenuItemsFilter />
|
<CanvasEntityMenuItemsFilter />
|
||||||
<CanvasEntityMenuItemsSelectObject />
|
<CanvasEntityMenuItemsSelectObject />
|
||||||
|
<RasterLayerMenuItemsAdjustments />
|
||||||
<MenuDivider />
|
<MenuDivider />
|
||||||
<CanvasEntityMenuItemsMergeDown />
|
<CanvasEntityMenuItemsMergeDown />
|
||||||
<RasterLayerMenuItemsCopyToSubMenu />
|
<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 { Flex } from '@invoke-ai/ui-library';
|
||||||
import { useStore } from '@nanostores/react';
|
import { useStore } from '@nanostores/react';
|
||||||
|
import { objectEquals } from '@observ33r/object-equals';
|
||||||
import { skipToken } from '@reduxjs/toolkit/query';
|
import { skipToken } from '@reduxjs/toolkit/query';
|
||||||
import { useAppSelector, useAppStore } from 'app/store/storeHooks';
|
import { useAppSelector, useAppStore } from 'app/store/storeHooks';
|
||||||
import { UploadImageIconButton } from 'common/hooks/useImageUploadButton';
|
import { UploadImageIconButton } from 'common/hooks/useImageUploadButton';
|
||||||
import { bboxSizeOptimized, bboxSizeRecalled } from 'features/controlLayers/store/canvasSlice';
|
import { bboxSizeOptimized, bboxSizeRecalled } from 'features/controlLayers/store/canvasSlice';
|
||||||
import { useCanvasIsStaging } from 'features/controlLayers/store/canvasStagingAreaSlice';
|
import { useCanvasIsStaging } from 'features/controlLayers/store/canvasStagingAreaSlice';
|
||||||
import { sizeOptimized, sizeRecalled } from 'features/controlLayers/store/paramsSlice';
|
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 type { setGlobalReferenceImageDndTarget, setRegionalGuidanceReferenceImageDndTarget } from 'features/dnd/dnd';
|
||||||
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
||||||
import { DndImage } from 'features/dnd/DndImage';
|
import { DndImage } from 'features/dnd/DndImage';
|
||||||
@@ -14,14 +18,14 @@ import { DndImageIcon } from 'features/dnd/DndImageIcon';
|
|||||||
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
||||||
import { memo, useCallback, useEffect } from 'react';
|
import { memo, useCallback, useEffect } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { PiArrowCounterClockwiseBold, PiRulerBold } from 'react-icons/pi';
|
import { PiArrowCounterClockwiseBold, PiCropBold, PiRulerBold } from 'react-icons/pi';
|
||||||
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
import { useGetImageDTOQuery, useUploadImageMutation } from 'services/api/endpoints/images';
|
||||||
import type { ImageDTO } from 'services/api/types';
|
import type { ImageDTO } from 'services/api/types';
|
||||||
import { $isConnected } from 'services/events/stores';
|
import { $isConnected } from 'services/events/stores';
|
||||||
|
|
||||||
type Props<T extends typeof setGlobalReferenceImageDndTarget | typeof setRegionalGuidanceReferenceImageDndTarget> = {
|
type Props<T extends typeof setGlobalReferenceImageDndTarget | typeof setRegionalGuidanceReferenceImageDndTarget> = {
|
||||||
image: ImageWithDims | null;
|
image: CroppableImageWithDims | null;
|
||||||
onChangeImage: (imageDTO: ImageDTO | null) => void;
|
onChangeImage: (croppableImage: CroppableImageWithDims | null) => void;
|
||||||
dndTarget: T;
|
dndTarget: T;
|
||||||
dndTargetData: ReturnType<T['getData']>;
|
dndTargetData: ReturnType<T['getData']>;
|
||||||
};
|
};
|
||||||
@@ -38,20 +42,28 @@ export const RefImageImage = memo(
|
|||||||
const isConnected = useStore($isConnected);
|
const isConnected = useStore($isConnected);
|
||||||
const tab = useAppSelector(selectActiveTab);
|
const tab = useAppSelector(selectActiveTab);
|
||||||
const isStaging = useCanvasIsStaging();
|
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(() => {
|
const handleResetControlImage = useCallback(() => {
|
||||||
onChangeImage(null);
|
onChangeImage(null);
|
||||||
}, [onChangeImage]);
|
}, [onChangeImage]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (isConnected && isError) {
|
if ((isConnected && croppedImageDTOReq.isError) || originalImageDTOReq.isError) {
|
||||||
handleResetControlImage();
|
handleResetControlImage();
|
||||||
}
|
}
|
||||||
}, [handleResetControlImage, isError, isConnected]);
|
}, [handleResetControlImage, isConnected, croppedImageDTOReq.isError, originalImageDTOReq.isError]);
|
||||||
|
|
||||||
const onUpload = useCallback(
|
const onUpload = useCallback(
|
||||||
(imageDTO: ImageDTO) => {
|
(imageDTO: ImageDTO) => {
|
||||||
onChangeImage(imageDTO);
|
onChangeImage(imageDTOToCroppableImage(imageDTO));
|
||||||
},
|
},
|
||||||
[onChangeImage]
|
[onChangeImage]
|
||||||
);
|
);
|
||||||
@@ -70,13 +82,67 @@ export const RefImageImage = memo(
|
|||||||
}
|
}
|
||||||
}, [imageDTO, isStaging, store, tab]);
|
}, [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 (
|
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 && (
|
{!imageDTO && (
|
||||||
<UploadImageIconButton
|
<UploadImageIconButton
|
||||||
w="full"
|
w="full"
|
||||||
h="full"
|
h="full"
|
||||||
isError={!imageDTO && !image?.image_name}
|
isError={!imageDTO && !imageWithDims?.image_name}
|
||||||
onUpload={onUpload}
|
onUpload={onUpload}
|
||||||
fontSize={36}
|
fontSize={36}
|
||||||
/>
|
/>
|
||||||
@@ -99,6 +165,15 @@ export const RefImageImage = memo(
|
|||||||
isDisabled={!imageDTO || (tab === 'canvas' && isStaging)}
|
isDisabled={!imageDTO || (tab === 'canvas' && isStaging)}
|
||||||
/>
|
/>
|
||||||
</Flex>
|
</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')} />
|
<DndDropTarget dndTarget={dndTarget} dndTargetData={dndTargetData} label={t('gallery.drop')} />
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
selectRefImageEntityIds,
|
selectRefImageEntityIds,
|
||||||
selectSelectedRefEntityId,
|
selectSelectedRefEntityId,
|
||||||
} from 'features/controlLayers/store/refImagesSlice';
|
} 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 { addGlobalReferenceImageDndTarget } from 'features/dnd/dnd';
|
||||||
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
||||||
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
||||||
@@ -92,7 +92,7 @@ const AddRefImageDropTargetAndButton = memo(() => {
|
|||||||
({
|
({
|
||||||
onUpload: (imageDTO: ImageDTO) => {
|
onUpload: (imageDTO: ImageDTO) => {
|
||||||
const config = getDefaultRefImageConfig(getState);
|
const config = getDefaultRefImageConfig(getState);
|
||||||
config.image = imageDTOToImageWithDims(imageDTO);
|
config.image = imageDTOToCroppableImage(imageDTO);
|
||||||
dispatch(refImageAdded({ overrides: { config } }));
|
dispatch(refImageAdded({ overrides: { config } }));
|
||||||
},
|
},
|
||||||
allowMultiple: false,
|
allowMultiple: false,
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import type { SystemStyleObject } from '@invoke-ai/ui-library';
|
import type { SystemStyleObject } from '@invoke-ai/ui-library';
|
||||||
import { Flex, Icon, IconButton, Image, Skeleton, Text, Tooltip } 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 { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
import { round } from 'es-toolkit/compat';
|
import { round } from 'es-toolkit/compat';
|
||||||
import { useRefImageEntity } from 'features/controlLayers/components/RefImage/useRefImageEntity';
|
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 { getGlobalReferenceImageWarnings } from 'features/controlLayers/store/validators';
|
||||||
import { memo, useCallback, useEffect, useMemo, useState } from 'react';
|
import { memo, useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
import { PiExclamationMarkBold, PiEyeSlashBold, PiImageBold } from 'react-icons/pi';
|
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';
|
import { RefImageWarningTooltipContent } from './RefImageWarningTooltipContent';
|
||||||
|
|
||||||
@@ -72,7 +71,8 @@ export const RefImagePreview = memo(() => {
|
|||||||
const selectedEntityId = useAppSelector(selectSelectedRefEntityId);
|
const selectedEntityId = useAppSelector(selectSelectedRefEntityId);
|
||||||
const isPanelOpen = useAppSelector(selectIsRefImagePanelOpen);
|
const isPanelOpen = useAppSelector(selectIsRefImagePanelOpen);
|
||||||
const [showWeightDisplay, setShowWeightDisplay] = useState(false);
|
const [showWeightDisplay, setShowWeightDisplay] = useState(false);
|
||||||
const { data: imageDTO } = useGetImageDTOQuery(entity.config.image?.image_name ?? skipToken);
|
|
||||||
|
const imageDTO = useImageDTOFromCroppableImage(entity.config.image);
|
||||||
|
|
||||||
const sx = useMemo(() => {
|
const sx = useMemo(() => {
|
||||||
if (!isIPAdapterConfig(entity.config)) {
|
if (!isIPAdapterConfig(entity.config)) {
|
||||||
@@ -145,7 +145,7 @@ export const RefImagePreview = memo(() => {
|
|||||||
overflow="hidden"
|
overflow="hidden"
|
||||||
>
|
>
|
||||||
<Image
|
<Image
|
||||||
src={imageDTO?.thumbnail_url}
|
src={imageDTO?.image_url}
|
||||||
objectFit="contain"
|
objectFit="contain"
|
||||||
aspectRatio="1/1"
|
aspectRatio="1/1"
|
||||||
height={imageDTO?.height}
|
height={imageDTO?.height}
|
||||||
|
|||||||
@@ -30,6 +30,7 @@ import {
|
|||||||
} from 'features/controlLayers/store/refImagesSlice';
|
} from 'features/controlLayers/store/refImagesSlice';
|
||||||
import type {
|
import type {
|
||||||
CLIPVisionModelV2,
|
CLIPVisionModelV2,
|
||||||
|
CroppableImageWithDims,
|
||||||
FLUXReduxImageInfluence as FLUXReduxImageInfluenceType,
|
FLUXReduxImageInfluence as FLUXReduxImageInfluenceType,
|
||||||
IPMethodV2,
|
IPMethodV2,
|
||||||
} from 'features/controlLayers/store/types';
|
} from 'features/controlLayers/store/types';
|
||||||
@@ -42,7 +43,6 @@ import type {
|
|||||||
ChatGPT4oModelConfig,
|
ChatGPT4oModelConfig,
|
||||||
FLUXKontextModelConfig,
|
FLUXKontextModelConfig,
|
||||||
FLUXReduxModelConfig,
|
FLUXReduxModelConfig,
|
||||||
ImageDTO,
|
|
||||||
IPAdapterModelConfig,
|
IPAdapterModelConfig,
|
||||||
} from 'services/api/types';
|
} from 'services/api/types';
|
||||||
|
|
||||||
@@ -104,15 +104,19 @@ const RefImageSettingsContent = memo(() => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const onChangeImage = useCallback(
|
const onChangeImage = useCallback(
|
||||||
(imageDTO: ImageDTO | null) => {
|
(croppableImage: CroppableImageWithDims | null) => {
|
||||||
dispatch(refImageImageChanged({ id, imageDTO }));
|
dispatch(refImageImageChanged({ id, croppableImage }));
|
||||||
},
|
},
|
||||||
[dispatch, id]
|
[dispatch, id]
|
||||||
);
|
);
|
||||||
|
|
||||||
const dndTargetData = useMemo<SetGlobalReferenceImageDndTargetData>(
|
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);
|
const isFLUX = useAppSelector(selectIsFLUX);
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import { FLUXReduxImageInfluence } from 'features/controlLayers/components/commo
|
|||||||
import { IPAdapterCLIPVisionModel } from 'features/controlLayers/components/common/IPAdapterCLIPVisionModel';
|
import { IPAdapterCLIPVisionModel } from 'features/controlLayers/components/common/IPAdapterCLIPVisionModel';
|
||||||
import { Weight } from 'features/controlLayers/components/common/Weight';
|
import { Weight } from 'features/controlLayers/components/common/Weight';
|
||||||
import { IPAdapterMethod } from 'features/controlLayers/components/RefImage/IPAdapterMethod';
|
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 { RegionalGuidanceIPAdapterSettingsEmptyState } from 'features/controlLayers/components/RegionalGuidance/RegionalGuidanceIPAdapterSettingsEmptyState';
|
||||||
import { RegionalReferenceImageModel } from 'features/controlLayers/components/RegionalGuidance/RegionalReferenceImageModel';
|
import { RegionalReferenceImageModel } from 'features/controlLayers/components/RegionalGuidance/RegionalReferenceImageModel';
|
||||||
import { useEntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
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 type { FLUXReduxModelConfig, ImageDTO, IPAdapterModelConfig } from 'services/api/types';
|
||||||
import { assert } from 'tsafe';
|
import { assert } from 'tsafe';
|
||||||
|
|
||||||
|
import { RegionalGuidanceRefImageImage } from './RegionalGuidanceRefImageImage';
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
referenceImageId: string;
|
referenceImageId: string;
|
||||||
};
|
};
|
||||||
@@ -114,7 +115,7 @@ const RegionalGuidanceIPAdapterSettingsContent = memo(({ referenceImageId }: Pro
|
|||||||
{ entityIdentifier, referenceImageId },
|
{ entityIdentifier, referenceImageId },
|
||||||
config.image?.image_name
|
config.image?.image_name
|
||||||
),
|
),
|
||||||
[entityIdentifier, config.image?.image_name, referenceImageId]
|
[entityIdentifier, config.image, referenceImageId]
|
||||||
);
|
);
|
||||||
|
|
||||||
const pullBboxIntoIPAdapter = usePullBboxIntoRegionalGuidanceReferenceImage(entityIdentifier, referenceImageId);
|
const pullBboxIntoIPAdapter = usePullBboxIntoRegionalGuidanceReferenceImage(entityIdentifier, referenceImageId);
|
||||||
@@ -170,7 +171,7 @@ const RegionalGuidanceIPAdapterSettingsContent = memo(({ referenceImageId }: Pro
|
|||||||
</Flex>
|
</Flex>
|
||||||
)}
|
)}
|
||||||
<Flex alignItems="center" justifyContent="center" h={32} w={32} aspectRatio="1/1" flexGrow={1}>
|
<Flex alignItems="center" justifyContent="center" h={32} w={32} aspectRatio="1/1" flexGrow={1}>
|
||||||
<RefImageImage
|
<RegionalGuidanceRefImageImage
|
||||||
image={config.image}
|
image={config.image}
|
||||||
onChangeImage={onChangeImage}
|
onChangeImage={onChangeImage}
|
||||||
dndTarget={setRegionalGuidanceReferenceImageDndTarget}
|
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 {
|
import { Flex, Heading, Spacer } from '@invoke-ai/ui-library';
|
||||||
Button,
|
|
||||||
ButtonGroup,
|
|
||||||
Flex,
|
|
||||||
Heading,
|
|
||||||
Icon,
|
|
||||||
ListItem,
|
|
||||||
Menu,
|
|
||||||
MenuButton,
|
|
||||||
MenuItem,
|
|
||||||
MenuList,
|
|
||||||
Spacer,
|
|
||||||
Spinner,
|
|
||||||
Text,
|
|
||||||
Tooltip,
|
|
||||||
UnorderedList,
|
|
||||||
} from '@invoke-ai/ui-library';
|
|
||||||
import { useStore } from '@nanostores/react';
|
import { useStore } from '@nanostores/react';
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
|
||||||
import { useFocusRegion, useIsRegionFocused } from 'common/hooks/focus';
|
import { useFocusRegion, useIsRegionFocused } from 'common/hooks/focus';
|
||||||
import { CanvasAutoProcessSwitch } from 'features/controlLayers/components/CanvasAutoProcessSwitch';
|
import { CanvasAutoProcessSwitch } from 'features/controlLayers/components/CanvasAutoProcessSwitch';
|
||||||
import { CanvasOperationIsolatedLayerPreviewSwitch } from 'features/controlLayers/components/CanvasOperationIsolatedLayerPreviewSwitch';
|
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 { SelectObjectInvert } from 'features/controlLayers/components/SelectObject/SelectObjectInvert';
|
||||||
import { SelectObjectPointType } from 'features/controlLayers/components/SelectObject/SelectObjectPointType';
|
import { SelectObjectPointType } from 'features/controlLayers/components/SelectObject/SelectObjectPointType';
|
||||||
import { useCanvasManager } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
import { useCanvasManager } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
||||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
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 { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||||
import type { PropsWithChildren } from 'react';
|
import { memo, useRef } from 'react';
|
||||||
import { memo, useCallback, useRef } from 'react';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { Trans, useTranslation } from 'react-i18next';
|
|
||||||
import { PiCaretDownBold, PiInfoBold } from 'react-icons/pi';
|
import { SelectObjectModel } from './SelectObjectModel';
|
||||||
|
import { SelectObjectPrompt } from './SelectObjectPrompt';
|
||||||
|
|
||||||
const SelectObjectContent = memo(
|
const SelectObjectContent = memo(
|
||||||
({ adapter }: { adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer }) => {
|
({ adapter }: { adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer }) => {
|
||||||
@@ -39,25 +25,7 @@ const SelectObjectContent = memo(
|
|||||||
useFocusRegion('canvas', ref, { focusOnMount: true });
|
useFocusRegion('canvas', ref, { focusOnMount: true });
|
||||||
const isCanvasFocused = useIsRegionFocused('canvas');
|
const isCanvasFocused = useIsRegionFocused('canvas');
|
||||||
const isProcessing = useStore(adapter.segmentAnything.$isProcessing);
|
const isProcessing = useStore(adapter.segmentAnything.$isProcessing);
|
||||||
const hasPoints = useStore(adapter.segmentAnything.$hasPoints);
|
const inputType = useStore(adapter.segmentAnything.$inputType);
|
||||||
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]);
|
|
||||||
|
|
||||||
useRegisteredHotkeys({
|
useRegisteredHotkeys({
|
||||||
id: 'applySegmentAnything',
|
id: 'applySegmentAnything',
|
||||||
@@ -94,11 +62,7 @@ const SelectObjectContent = memo(
|
|||||||
<Heading size="md" color="base.300" userSelect="none">
|
<Heading size="md" color="base.300" userSelect="none">
|
||||||
{t('controlLayers.selectObject.selectObject')}
|
{t('controlLayers.selectObject.selectObject')}
|
||||||
</Heading>
|
</Heading>
|
||||||
<Tooltip label={<SelectObjectHelpTooltipContent />}>
|
<SelectObjectInfoTooltip />
|
||||||
<Flex alignItems="center">
|
|
||||||
<Icon as={PiInfoBold} color="base.500" />
|
|
||||||
</Flex>
|
|
||||||
</Tooltip>
|
|
||||||
</Flex>
|
</Flex>
|
||||||
<Spacer />
|
<Spacer />
|
||||||
<CanvasAutoProcessSwitch />
|
<CanvasAutoProcessSwitch />
|
||||||
@@ -106,71 +70,14 @@ const SelectObjectContent = memo(
|
|||||||
</Flex>
|
</Flex>
|
||||||
|
|
||||||
<Flex w="full" justifyContent="space-between" py={2}>
|
<Flex w="full" justifyContent="space-between" py={2}>
|
||||||
<SelectObjectPointType adapter={adapter} />
|
<SelectObjectInputTypeButtons adapter={adapter} />
|
||||||
<SelectObjectInvert adapter={adapter} />
|
<SelectObjectInvert adapter={adapter} />
|
||||||
</Flex>
|
</Flex>
|
||||||
|
|
||||||
<ButtonGroup isAttached={false} size="sm" w="full">
|
{inputType === 'visual' && <SelectObjectPointType adapter={adapter} />}
|
||||||
<Button
|
{inputType === 'prompt' && <SelectObjectPrompt adapter={adapter} />}
|
||||||
onClick={adapter.segmentAnything.processImmediate}
|
<SelectObjectModel adapter={adapter} />
|
||||||
loadingText={t('controlLayers.selectObject.process')}
|
<SelectObjectActionButtons adapter={adapter} />
|
||||||
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>
|
|
||||||
</Flex>
|
</Flex>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -190,34 +97,3 @@ export const SelectObject = memo(() => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
SelectObject.displayName = 'SelectObject';
|
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">
|
<FormControl w="min-content">
|
||||||
<FormLabel m={0}>{t('controlLayers.selectObject.pointType')}</FormLabel>
|
<FormLabel m={0}>{t('controlLayers.selectObject.pointType')}</FormLabel>
|
||||||
<RadioGroup value={pointType} onChange={onChange} w="full" size="md">
|
<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">
|
<Radio value="foreground">
|
||||||
<Text>{t('controlLayers.selectObject.include')}</Text>
|
<Text>{t('controlLayers.selectObject.include')}</Text>
|
||||||
</Radio>
|
</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,
|
PopoverBody,
|
||||||
PopoverContent,
|
PopoverContent,
|
||||||
PopoverTrigger,
|
PopoverTrigger,
|
||||||
|
Portal,
|
||||||
Text,
|
Text,
|
||||||
useShiftModifier,
|
useShiftModifier,
|
||||||
} from '@invoke-ai/ui-library';
|
} from '@invoke-ai/ui-library';
|
||||||
@@ -45,62 +46,64 @@ export const CanvasSettingsPopover = memo(() => {
|
|||||||
alignSelf="stretch"
|
alignSelf="stretch"
|
||||||
/>
|
/>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent maxW="280px">
|
<Portal>
|
||||||
<PopoverArrow />
|
<PopoverContent maxW="280px">
|
||||||
<PopoverBody>
|
<PopoverArrow />
|
||||||
<Flex direction="column" gap={2}>
|
<PopoverBody>
|
||||||
{/* Behavior Settings */}
|
<Flex direction="column" gap={2}>
|
||||||
<Flex direction="column" gap={1}>
|
{/* Behavior Settings */}
|
||||||
<Flex align="center" gap={2}>
|
<Flex direction="column" gap={1}>
|
||||||
<Icon as={PiPencilFill} boxSize={4} />
|
<Flex align="center" gap={2}>
|
||||||
<Text fontWeight="bold" fontSize="sm" color="base.100">
|
<Icon as={PiPencilFill} boxSize={4} />
|
||||||
{t('hotkeys.canvas.settings.behavior')}
|
<Text fontWeight="bold" fontSize="sm" color="base.100">
|
||||||
</Text>
|
{t('hotkeys.canvas.settings.behavior')}
|
||||||
|
</Text>
|
||||||
|
</Flex>
|
||||||
|
<CanvasSettingsInvertScrollCheckbox />
|
||||||
|
<CanvasSettingsPressureSensitivityCheckbox />
|
||||||
|
<CanvasSettingsPreserveMaskCheckbox />
|
||||||
|
<CanvasSettingsClipToBboxCheckbox />
|
||||||
|
<CanvasSettingsOutputOnlyMaskedRegionsCheckbox />
|
||||||
|
<CanvasSettingsSaveAllImagesToGalleryCheckbox />
|
||||||
</Flex>
|
</Flex>
|
||||||
<CanvasSettingsInvertScrollCheckbox />
|
|
||||||
<CanvasSettingsPressureSensitivityCheckbox />
|
|
||||||
<CanvasSettingsPreserveMaskCheckbox />
|
|
||||||
<CanvasSettingsClipToBboxCheckbox />
|
|
||||||
<CanvasSettingsOutputOnlyMaskedRegionsCheckbox />
|
|
||||||
<CanvasSettingsSaveAllImagesToGalleryCheckbox />
|
|
||||||
</Flex>
|
|
||||||
|
|
||||||
<Divider />
|
<Divider />
|
||||||
|
|
||||||
{/* Display Settings */}
|
{/* Display Settings */}
|
||||||
<Flex direction="column" gap={1}>
|
<Flex direction="column" gap={1}>
|
||||||
<Flex align="center" gap={2} color="base.200">
|
<Flex align="center" gap={2} color="base.200">
|
||||||
<Icon as={PiEyeFill} boxSize={4} />
|
<Icon as={PiEyeFill} boxSize={4} />
|
||||||
<Text fontWeight="bold" fontSize="sm">
|
<Text fontWeight="bold" fontSize="sm">
|
||||||
{t('hotkeys.canvas.settings.display')}
|
{t('hotkeys.canvas.settings.display')}
|
||||||
</Text>
|
</Text>
|
||||||
|
</Flex>
|
||||||
|
<CanvasSettingsShowProgressOnCanvas />
|
||||||
|
<CanvasSettingsIsolatedStagingPreviewSwitch />
|
||||||
|
<CanvasSettingsIsolatedLayerPreviewSwitch />
|
||||||
|
<CanvasSettingsBboxOverlaySwitch />
|
||||||
|
<CanvasSettingsShowHUDSwitch />
|
||||||
</Flex>
|
</Flex>
|
||||||
<CanvasSettingsShowProgressOnCanvas />
|
|
||||||
<CanvasSettingsIsolatedStagingPreviewSwitch />
|
|
||||||
<CanvasSettingsIsolatedLayerPreviewSwitch />
|
|
||||||
<CanvasSettingsBboxOverlaySwitch />
|
|
||||||
<CanvasSettingsShowHUDSwitch />
|
|
||||||
</Flex>
|
|
||||||
|
|
||||||
<Divider />
|
<Divider />
|
||||||
|
|
||||||
{/* Grid Settings */}
|
{/* Grid Settings */}
|
||||||
<Flex direction="column" gap={1}>
|
<Flex direction="column" gap={1}>
|
||||||
<Flex align="center" gap={2} color="base.200">
|
<Flex align="center" gap={2} color="base.200">
|
||||||
<Icon as={PiSquaresFourFill} boxSize={4} />
|
<Icon as={PiSquaresFourFill} boxSize={4} />
|
||||||
<Text fontWeight="bold" fontSize="sm">
|
<Text fontWeight="bold" fontSize="sm">
|
||||||
{t('hotkeys.canvas.settings.grid')}
|
{t('hotkeys.canvas.settings.grid')}
|
||||||
</Text>
|
</Text>
|
||||||
|
</Flex>
|
||||||
|
<CanvasSettingsSnapToGridCheckbox />
|
||||||
|
<CanvasSettingsDynamicGridSwitch />
|
||||||
|
<CanvasSettingsRuleOfThirdsSwitch />
|
||||||
</Flex>
|
</Flex>
|
||||||
<CanvasSettingsSnapToGridCheckbox />
|
|
||||||
<CanvasSettingsDynamicGridSwitch />
|
|
||||||
<CanvasSettingsRuleOfThirdsSwitch />
|
|
||||||
</Flex>
|
|
||||||
|
|
||||||
<DebugSettings />
|
<DebugSettings />
|
||||||
</Flex>
|
</Flex>
|
||||||
</PopoverBody>
|
</PopoverBody>
|
||||||
</PopoverContent>
|
</PopoverContent>
|
||||||
|
</Portal>
|
||||||
</Popover>
|
</Popover>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -707,10 +707,9 @@ describe('StagingAreaApi', () => {
|
|||||||
|
|
||||||
// Should end up with the last set of items
|
// Should end up with the last set of items
|
||||||
expect(api.$items.get()).toBe(items2);
|
expect(api.$items.get()).toBe(items2);
|
||||||
// The selectedItemId retains the old value (1) but $selectedItem will be null
|
// We expect the selection to have moved to the next existent item
|
||||||
// because item 1 is no longer in the items list
|
expect(api.$selectedItemId.get()).toBe(2);
|
||||||
expect(api.$selectedItemId.get()).toBe(1);
|
expect(api.$selectedItem.get()?.item.item_id).toBe(2);
|
||||||
expect(api.$selectedItem.get()).toBe(null);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should handle multiple progress events for same item', () => {
|
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);
|
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,35 +6,65 @@ import {
|
|||||||
PopoverBody,
|
PopoverBody,
|
||||||
PopoverContent,
|
PopoverContent,
|
||||||
PopoverTrigger,
|
PopoverTrigger,
|
||||||
|
Portal,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
} from '@invoke-ai/ui-library';
|
} from '@invoke-ai/ui-library';
|
||||||
import { createSelector } from '@reduxjs/toolkit';
|
import { createSelector } from '@reduxjs/toolkit';
|
||||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
import RgbaColorPicker from 'common/components/ColorPicker/RgbaColorPicker';
|
import RgbaColorPicker from 'common/components/ColorPicker/RgbaColorPicker';
|
||||||
import { rgbaColorToString } from 'common/util/colorCodeTransformers';
|
import { rgbaColorToString } from 'common/util/colorCodeTransformers';
|
||||||
import { selectCanvasSettingsSlice, settingsColorChanged } from 'features/controlLayers/store/canvasSettingsSlice';
|
import {
|
||||||
|
selectCanvasSettingsSlice,
|
||||||
|
settingsActiveColorToggled,
|
||||||
|
settingsBgColorChanged,
|
||||||
|
settingsColorsSetToDefault,
|
||||||
|
settingsFgColorChanged,
|
||||||
|
} from 'features/controlLayers/store/canvasSettingsSlice';
|
||||||
import type { RgbaColor } from 'features/controlLayers/store/types';
|
import type { RgbaColor } from 'features/controlLayers/store/types';
|
||||||
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||||
import { memo, useCallback } from 'react';
|
import { memo, useCallback, useMemo } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
|
||||||
const selectColor = createSelector(selectCanvasSettingsSlice, (settings) => settings.color);
|
const selectActiveColor = createSelector(selectCanvasSettingsSlice, (settings) => settings.activeColor);
|
||||||
|
const selectBgColor = createSelector(selectCanvasSettingsSlice, (settings) => settings.bgColor);
|
||||||
|
const selectFgColor = createSelector(selectCanvasSettingsSlice, (settings) => settings.fgColor);
|
||||||
|
|
||||||
export const ToolColorPicker = memo(() => {
|
export const ToolFillColorPicker = memo(() => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const fill = useAppSelector(selectColor);
|
const activeColorType = useAppSelector(selectActiveColor);
|
||||||
|
const bgColor = useAppSelector(selectBgColor);
|
||||||
|
const fgColor = useAppSelector(selectFgColor);
|
||||||
|
const { activeColor, tooltip, bgColorzIndex, fgColorzIndex } = useMemo(() => {
|
||||||
|
if (activeColorType === 'bgColor') {
|
||||||
|
return { activeColor: bgColor, tooltip: t('controlLayers.fill.bgFillColor'), bgColorzIndex: 2, fgColorzIndex: 1 };
|
||||||
|
} else {
|
||||||
|
return { activeColor: fgColor, tooltip: t('controlLayers.fill.fgFillColor'), bgColorzIndex: 1, fgColorzIndex: 2 };
|
||||||
|
}
|
||||||
|
}, [activeColorType, bgColor, fgColor, t]);
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const onChange = useCallback(
|
const onColorChange = useCallback(
|
||||||
(color: RgbaColor) => {
|
(color: RgbaColor) => {
|
||||||
dispatch(settingsColorChanged(color));
|
if (activeColorType === 'bgColor') {
|
||||||
|
dispatch(settingsBgColorChanged(color));
|
||||||
|
} else {
|
||||||
|
dispatch(settingsFgColorChanged(color));
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[dispatch]
|
[activeColorType, dispatch]
|
||||||
);
|
);
|
||||||
|
|
||||||
useRegisteredHotkeys({
|
useRegisteredHotkeys({
|
||||||
id: 'setFillToWhite',
|
id: 'setFillColorsToDefault',
|
||||||
category: 'canvas',
|
category: 'canvas',
|
||||||
callback: () => dispatch(settingsColorChanged({ r: 255, g: 255, b: 255, a: 1 })),
|
callback: () => dispatch(settingsColorsSetToDefault()),
|
||||||
|
options: { preventDefault: true },
|
||||||
|
dependencies: [dispatch],
|
||||||
|
});
|
||||||
|
|
||||||
|
useRegisteredHotkeys({
|
||||||
|
id: 'toggleFillColor',
|
||||||
|
category: 'canvas',
|
||||||
|
callback: () => dispatch(settingsActiveColorToggled()),
|
||||||
options: { preventDefault: true },
|
options: { preventDefault: true },
|
||||||
dependencies: [dispatch],
|
dependencies: [dispatch],
|
||||||
});
|
});
|
||||||
@@ -42,29 +72,47 @@ export const ToolColorPicker = memo(() => {
|
|||||||
return (
|
return (
|
||||||
<Popover isLazy>
|
<Popover isLazy>
|
||||||
<PopoverTrigger>
|
<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={t('controlLayers.fill.fillColor')}>
|
<Tooltip label={tooltip}>
|
||||||
<Flex w="full" h="full" alignItems="center" justifyContent="center">
|
<Flex alignItems="center" justifyContent="center" position="relative" w="full" h="full">
|
||||||
<Box
|
<Box
|
||||||
borderRadius="full"
|
borderRadius="full"
|
||||||
borderColor="base.600"
|
borderColor="base.600"
|
||||||
w={6}
|
w={6}
|
||||||
h={6}
|
h={6}
|
||||||
borderWidth={2}
|
borderWidth={2}
|
||||||
bg={rgbaColorToString(fill)}
|
bg={rgbaColorToString(bgColor)}
|
||||||
|
position="absolute"
|
||||||
|
top="0"
|
||||||
|
left="0"
|
||||||
|
zIndex={bgColorzIndex}
|
||||||
|
/>
|
||||||
|
<Box
|
||||||
|
borderRadius="full"
|
||||||
|
borderColor="base.600"
|
||||||
|
w={6}
|
||||||
|
h={6}
|
||||||
|
borderWidth={2}
|
||||||
|
bg={rgbaColorToString(fgColor)}
|
||||||
|
position="absolute"
|
||||||
|
top="2"
|
||||||
|
left="2"
|
||||||
|
zIndex={fgColorzIndex}
|
||||||
/>
|
/>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Flex>
|
</Flex>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent>
|
<Portal>
|
||||||
<PopoverArrow />
|
<PopoverContent>
|
||||||
<PopoverBody minH={64}>
|
<PopoverArrow />
|
||||||
<RgbaColorPicker color={fill} onChange={onChange} withNumberInput withSwatches />
|
<PopoverBody minH={64}>
|
||||||
</PopoverBody>
|
<RgbaColorPicker color={activeColor} onChange={onColorChange} withNumberInput withSwatches />
|
||||||
</PopoverContent>
|
</PopoverBody>
|
||||||
|
</PopoverContent>
|
||||||
|
</Portal>
|
||||||
</Popover>
|
</Popover>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
ToolColorPicker.displayName = 'ToolFillColorPicker';
|
ToolFillColorPicker.displayName = 'ToolFillColorPicker';
|
||||||
|
|||||||
@@ -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';
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user