|
|
@ -8,7 +8,7 @@ import { useRouter } from 'next/router'
|
|
|
|
import { AnimatePresence, motion } from 'framer-motion'
|
|
|
|
import { AnimatePresence, motion } from 'framer-motion'
|
|
|
|
import { HexAlphaColorPicker, HexColorInput } from 'react-colorful'
|
|
|
|
import { HexAlphaColorPicker, HexColorInput } from 'react-colorful'
|
|
|
|
import { MdFileUpload, MdHelpOutline } from 'react-icons/md'
|
|
|
|
import { MdFileUpload, MdHelpOutline } from 'react-icons/md'
|
|
|
|
import { FaQuestion } from 'react-icons/fa'
|
|
|
|
import { FaCoffee, FaQuestion } from 'react-icons/fa'
|
|
|
|
import TextareaAutosize from 'react-textarea-autosize'
|
|
|
|
import TextareaAutosize from 'react-textarea-autosize'
|
|
|
|
import JSZip from 'jszip'
|
|
|
|
import JSZip from 'jszip'
|
|
|
|
import { saveAs } from 'file-saver'
|
|
|
|
import { saveAs } from 'file-saver'
|
|
|
@ -64,7 +64,7 @@ import Checkbox from '../components/controls/checkbox'
|
|
|
|
import { getStorage } from '../data/storage/storage'
|
|
|
|
import { getStorage } from '../data/storage/storage'
|
|
|
|
import { getUserSetting } from '../data/user/settings'
|
|
|
|
import { getUserSetting } from '../data/user/settings'
|
|
|
|
import { useReload } from '../hooks/useReload'
|
|
|
|
import { useReload } from '../hooks/useReload'
|
|
|
|
import { BackendURLTagSearch } from '../globals/constants'
|
|
|
|
import { BackendURLTagSearch, IsIsekai } from '../globals/constants'
|
|
|
|
import Tooltip from '../components/tooltip'
|
|
|
|
import Tooltip from '../components/tooltip'
|
|
|
|
import { WorkerInterface } from '../tokenizer/interface'
|
|
|
|
import { WorkerInterface } from '../tokenizer/interface'
|
|
|
|
import { EncoderType } from '../tokenizer/encoder'
|
|
|
|
import { EncoderType } from '../tokenizer/encoder'
|
|
|
@ -561,7 +561,7 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
|
|
|
|
|
|
|
|
const [purchaseModalOpen, setPurchaseModalOpen] = useState(false)
|
|
|
|
const [purchaseModalOpen, setPurchaseModalOpen] = useState(false)
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {}, [
|
|
|
|
useEffect(() => { }, [
|
|
|
|
params,
|
|
|
|
params,
|
|
|
|
selectedModel,
|
|
|
|
selectedModel,
|
|
|
|
session,
|
|
|
|
session,
|
|
|
@ -604,10 +604,10 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
lastPromptTokensGood.current = false
|
|
|
|
lastPromptTokensGood.current = false
|
|
|
|
toast.warn(
|
|
|
|
toast.warn(
|
|
|
|
'Prompt is too long and will be cut off. Using ' +
|
|
|
|
'Prompt is too long and will be cut off. Using ' +
|
|
|
|
promptTokens +
|
|
|
|
promptTokens +
|
|
|
|
' out of ' +
|
|
|
|
' out of ' +
|
|
|
|
SD_TOKEN_LIMIT +
|
|
|
|
SD_TOKEN_LIMIT +
|
|
|
|
' available tokens.'
|
|
|
|
' available tokens.'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
} else if (promptTokens.every((t) => t <= SD_TOKEN_LIMIT)) {
|
|
|
|
} else if (promptTokens.every((t) => t <= SD_TOKEN_LIMIT)) {
|
|
|
|
lastPromptTokensGood.current = true
|
|
|
|
lastPromptTokensGood.current = true
|
|
|
@ -646,10 +646,10 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
lastNegPromptTokensGood.current = false
|
|
|
|
lastNegPromptTokensGood.current = false
|
|
|
|
toast.warn(
|
|
|
|
toast.warn(
|
|
|
|
'Negative prompt is too long and will be cut off. Using ' +
|
|
|
|
'Negative prompt is too long and will be cut off. Using ' +
|
|
|
|
negPromptTokens +
|
|
|
|
negPromptTokens +
|
|
|
|
' out of ' +
|
|
|
|
' out of ' +
|
|
|
|
SD_TOKEN_LIMIT +
|
|
|
|
SD_TOKEN_LIMIT +
|
|
|
|
' available tokens.'
|
|
|
|
' available tokens.'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
} else if (negPromptTokens <= SD_TOKEN_LIMIT) {
|
|
|
|
} else if (negPromptTokens <= SD_TOKEN_LIMIT) {
|
|
|
|
lastNegPromptTokensGood.current = true
|
|
|
|
lastNegPromptTokensGood.current = true
|
|
|
@ -673,8 +673,8 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
lastTagSearchRequest.current = requestId
|
|
|
|
lastTagSearchRequest.current = requestId
|
|
|
|
fetch(
|
|
|
|
fetch(
|
|
|
|
BackendURLTagSearch +
|
|
|
|
BackendURLTagSearch +
|
|
|
|
//`?model=${encodeURIComponent(selectedModel.toString())}&prompt=${encodeURIComponent(
|
|
|
|
//`?model=${encodeURIComponent(selectedModel.toString())}&prompt=${encodeURIComponent(
|
|
|
|
`?prompt=${encodeURIComponent(prompt.trim())}`,
|
|
|
|
`?prompt=${encodeURIComponent(prompt.trim())}`,
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mode: 'cors',
|
|
|
|
mode: 'cors',
|
|
|
|
cache: 'default',
|
|
|
|
cache: 'default',
|
|
|
@ -2004,18 +2004,43 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
<HistoryIcon />
|
|
|
|
<HistoryIcon />
|
|
|
|
</OpenHistoryButton>
|
|
|
|
</OpenHistoryButton>
|
|
|
|
<FlexSpaceFull />
|
|
|
|
<FlexSpaceFull />
|
|
|
|
<ImportImageLink
|
|
|
|
<div
|
|
|
|
href="//docs.novelai.net"
|
|
|
|
|
|
|
|
target="_blank"
|
|
|
|
|
|
|
|
style={{
|
|
|
|
style={{
|
|
|
|
fontSize: '0.875rem',
|
|
|
|
display: 'flex',
|
|
|
|
opacity: 0.8,
|
|
|
|
gap: '10px',
|
|
|
|
width: 'max-content',
|
|
|
|
alignItems: 'center',
|
|
|
|
padding: '0 0 0 10px',
|
|
|
|
justifyContent: 'space-between'
|
|
|
|
}}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
>
|
|
|
|
<FaQuestion />
|
|
|
|
{IsIsekai && (
|
|
|
|
</ImportImageLink>
|
|
|
|
<ImportImageLink
|
|
|
|
|
|
|
|
href="https://www.isekai.cn"
|
|
|
|
|
|
|
|
target="_blank"
|
|
|
|
|
|
|
|
style={{
|
|
|
|
|
|
|
|
fontSize: '0.875rem',
|
|
|
|
|
|
|
|
opacity: 0.8,
|
|
|
|
|
|
|
|
width: 'max-content',
|
|
|
|
|
|
|
|
padding: '0 0 0 10px',
|
|
|
|
|
|
|
|
}}
|
|
|
|
|
|
|
|
>
|
|
|
|
|
|
|
|
<FaCoffee />
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
异世界百科
|
|
|
|
|
|
|
|
</ImportImageLink>
|
|
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
<ImportImageLink
|
|
|
|
|
|
|
|
href="//docs.novelai.net"
|
|
|
|
|
|
|
|
target="_blank"
|
|
|
|
|
|
|
|
style={{
|
|
|
|
|
|
|
|
fontSize: '0.875rem',
|
|
|
|
|
|
|
|
opacity: 0.8,
|
|
|
|
|
|
|
|
width: 'max-content',
|
|
|
|
|
|
|
|
padding: '0 0 0 10px',
|
|
|
|
|
|
|
|
}}
|
|
|
|
|
|
|
|
>
|
|
|
|
|
|
|
|
<FaQuestion />
|
|
|
|
|
|
|
|
</ImportImageLink>
|
|
|
|
|
|
|
|
</div>
|
|
|
|
</HideMobileInline>
|
|
|
|
</HideMobileInline>
|
|
|
|
</MainTopperInnerRight>
|
|
|
|
</MainTopperInnerRight>
|
|
|
|
</MainTopper>
|
|
|
|
</MainTopper>
|
|
|
@ -2244,12 +2269,12 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
return e.key === 'ArrowUp'
|
|
|
|
return e.key === 'ArrowUp'
|
|
|
|
? (suggestionSelectionIndex +
|
|
|
|
? (suggestionSelectionIndex +
|
|
|
|
tagSuggestions.length -
|
|
|
|
tagSuggestions.length -
|
|
|
|
1) %
|
|
|
|
1) %
|
|
|
|
tagSuggestions.length
|
|
|
|
tagSuggestions.length
|
|
|
|
: (suggestionSelectionIndex +
|
|
|
|
: (suggestionSelectionIndex +
|
|
|
|
1) %
|
|
|
|
1) %
|
|
|
|
tagSuggestions.length
|
|
|
|
tagSuggestions.length
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
@ -2267,7 +2292,7 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
e.stopPropagation()
|
|
|
|
e.stopPropagation()
|
|
|
|
insertTagToPrompt(
|
|
|
|
insertTagToPrompt(
|
|
|
|
tagSuggestions[
|
|
|
|
tagSuggestions[
|
|
|
|
suggestionSelectionIndex
|
|
|
|
suggestionSelectionIndex
|
|
|
|
][0],
|
|
|
|
][0],
|
|
|
|
promptid
|
|
|
|
promptid
|
|
|
|
)
|
|
|
|
)
|
|
|
@ -2322,11 +2347,10 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
SD_TOKEN_LIMIT
|
|
|
|
SD_TOKEN_LIMIT
|
|
|
|
}
|
|
|
|
}
|
|
|
|
style={{
|
|
|
|
style={{
|
|
|
|
height: `${
|
|
|
|
height: `${((promptTokens[promptid] ?? 0) /
|
|
|
|
((promptTokens[promptid] ?? 0) /
|
|
|
|
SD_TOKEN_LIMIT) *
|
|
|
|
SD_TOKEN_LIMIT) *
|
|
|
|
|
|
|
|
100
|
|
|
|
100
|
|
|
|
}%`,
|
|
|
|
}%`,
|
|
|
|
}}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
</TokenLimitBarOuter>
|
|
|
|
</TokenLimitBarOuter>
|
|
|
@ -2483,19 +2507,19 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
{(shownHover === i ||
|
|
|
|
{(shownHover === i ||
|
|
|
|
images[selectedImage].length === 1) && (
|
|
|
|
images[selectedImage].length === 1) && (
|
|
|
|
<>
|
|
|
|
<>
|
|
|
|
{images[selectedImage].length === 1 &&
|
|
|
|
{images[selectedImage].length === 1 &&
|
|
|
|
img &&
|
|
|
|
img &&
|
|
|
|
topButtonGroup(img, 0)}
|
|
|
|
topButtonGroup(img, 0)}
|
|
|
|
|
|
|
|
|
|
|
|
{images[selectedImage].length !== 1 &&
|
|
|
|
{images[selectedImage].length !== 1 &&
|
|
|
|
specialButtonGroup(img, i)}
|
|
|
|
specialButtonGroup(img, i)}
|
|
|
|
{saveButtonGroup(img)}
|
|
|
|
{saveButtonGroup(img)}
|
|
|
|
|
|
|
|
|
|
|
|
{images[selectedImage].length === 1 &&
|
|
|
|
{images[selectedImage].length === 1 &&
|
|
|
|
seedButtonGroup(img, i)}
|
|
|
|
seedButtonGroup(img, i)}
|
|
|
|
</>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
)}
|
|
|
|
{img.isVariationOriginal && (
|
|
|
|
{img.isVariationOriginal && (
|
|
|
|
<div
|
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
style={{
|
|
|
@ -2585,9 +2609,9 @@ function ImageGenContent(): JSX.Element {
|
|
|
|
src={
|
|
|
|
src={
|
|
|
|
showOld === expandedImage
|
|
|
|
showOld === expandedImage
|
|
|
|
? images[selectedImage][expandedImage]
|
|
|
|
? images[selectedImage][expandedImage]
|
|
|
|
.enhanceOriginal
|
|
|
|
.enhanceOriginal
|
|
|
|
: images[selectedImage][expandedImage]
|
|
|
|
: images[selectedImage][expandedImage]
|
|
|
|
.url
|
|
|
|
.url
|
|
|
|
}
|
|
|
|
}
|
|
|
|
alt={images[selectedImage][
|
|
|
|
alt={images[selectedImage][
|
|
|
|
expandedImage
|
|
|
|
expandedImage
|
|
|
@ -3051,7 +3075,7 @@ const ExpandedImageContainer = styled(motion.div)`
|
|
|
|
align-items: center;
|
|
|
|
align-items: center;
|
|
|
|
`
|
|
|
|
`
|
|
|
|
|
|
|
|
|
|
|
|
const HistoryButton = styled(SubtleButton)<{ selected: boolean; img: string }>`
|
|
|
|
const HistoryButton = styled(SubtleButton) <{ selected: boolean; img: string }>`
|
|
|
|
border: ${(props) =>
|
|
|
|
border: ${(props) =>
|
|
|
|
props.selected ? `2px solid ${props.theme.colors.textHeadings}` : `2px solid transparent`};
|
|
|
|
props.selected ? `2px solid ${props.theme.colors.textHeadings}` : `2px solid transparent`};
|
|
|
|
border-radius: 3px;
|
|
|
|
border-radius: 3px;
|
|
|
@ -3323,7 +3347,7 @@ function GenerationOptions(props: {
|
|
|
|
onMouseDown: (e: any) => {
|
|
|
|
onMouseDown: (e: any) => {
|
|
|
|
if (e.target !== document.activeElement) {
|
|
|
|
if (e.target !== document.activeElement) {
|
|
|
|
e.preventDefault()
|
|
|
|
e.preventDefault()
|
|
|
|
;(e.target as HTMLInputElement).select()
|
|
|
|
; (e.target as HTMLInputElement).select()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -3692,7 +3716,6 @@ function GenerationOptions(props: {
|
|
|
|
maxRows={6}
|
|
|
|
maxRows={6}
|
|
|
|
placeholder="Anything in here is added to the preset selected above."
|
|
|
|
placeholder="Anything in here is added to the preset selected above."
|
|
|
|
value={props.negPrompt ?? ''}
|
|
|
|
value={props.negPrompt ?? ''}
|
|
|
|
warn={props.negPromptTokens > SD_TOKEN_LIMIT}
|
|
|
|
|
|
|
|
onChange={(e) => {
|
|
|
|
onChange={(e) => {
|
|
|
|
props.setNegPrompt((e.target.value || '').replace(/(\n|\r)/g, ''))
|
|
|
|
props.setNegPrompt((e.target.value || '').replace(/(\n|\r)/g, ''))
|
|
|
|
}}
|
|
|
|
}}
|
|
|
@ -3888,7 +3911,7 @@ const Input = styled.input<{ warn?: boolean }>`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
`
|
|
|
|
|
|
|
|
|
|
|
|
const LargeInput = styled(TextareaAutosize)<{ warn?: boolean }>`
|
|
|
|
const LargeInput = styled(TextareaAutosize) <{ warn?: boolean }>`
|
|
|
|
padding: 12px 20px;
|
|
|
|
padding: 12px 20px;
|
|
|
|
border: 1px solid ${(props) => (props.warn ? props.theme.colors.warning : props.theme.colors.bg3)};
|
|
|
|
border: 1px solid ${(props) => (props.warn ? props.theme.colors.warning : props.theme.colors.bg3)};
|
|
|
|
border-radius: 3px;
|
|
|
|
border-radius: 3px;
|
|
|
@ -4907,7 +4930,7 @@ function Canvas(props: {
|
|
|
|
onMouseDown: (e: any) => {
|
|
|
|
onMouseDown: (e: any) => {
|
|
|
|
if (e.target !== document.activeElement) {
|
|
|
|
if (e.target !== document.activeElement) {
|
|
|
|
e.preventDefault()
|
|
|
|
e.preventDefault()
|
|
|
|
;(e.target as HTMLInputElement).select()
|
|
|
|
; (e.target as HTMLInputElement).select()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -5537,7 +5560,7 @@ function Canvas(props: {
|
|
|
|
backgroundImage: props.rerollImage
|
|
|
|
backgroundImage: props.rerollImage
|
|
|
|
? `url(${props.rerollImage?.url})`
|
|
|
|
? `url(${props.rerollImage?.url})`
|
|
|
|
: // eslint-disable-next-line max-len
|
|
|
|
: // eslint-disable-next-line max-len
|
|
|
|
`linear-gradient(45deg, #A0A0A0 25%, transparent 25%), linear-gradient(-45deg, #A0A0A0 25%, transparent 25%), linear-gradient(45deg, transparent 75%, #A0A0A0 75%), linear-gradient(-45deg, transparent 75%, #A0A0A0 75%)`,
|
|
|
|
`linear-gradient(45deg, #A0A0A0 25%, transparent 25%), linear-gradient(-45deg, #A0A0A0 25%, transparent 25%), linear-gradient(45deg, transparent 75%, #A0A0A0 75%), linear-gradient(-45deg, transparent 75%, #A0A0A0 75%)`,
|
|
|
|
backgroundSize: props.rerollImage?.url ? '100%' : '30px 30px',
|
|
|
|
backgroundSize: props.rerollImage?.url ? '100%' : '30px 30px',
|
|
|
|
backgroundPosition: '0 0, 0 15px, 15px -15px, -15px 0px',
|
|
|
|
backgroundPosition: '0 0, 0 15px, 15px -15px, -15px 0px',
|
|
|
|
}}
|
|
|
|
}}
|
|
|
@ -5882,7 +5905,7 @@ const Filler = styled.div`
|
|
|
|
height: 30px;
|
|
|
|
height: 30px;
|
|
|
|
`
|
|
|
|
`
|
|
|
|
|
|
|
|
|
|
|
|
const CanvasControlButton = styled(SubtleButton)<{ selected?: boolean }>`
|
|
|
|
const CanvasControlButton = styled(SubtleButton) <{ selected?: boolean }>`
|
|
|
|
padding: 8px;
|
|
|
|
padding: 8px;
|
|
|
|
height: 44px;
|
|
|
|
height: 44px;
|
|
|
|
min-width: 58px;
|
|
|
|
min-width: 58px;
|
|
|
@ -5902,13 +5925,13 @@ const CanvasControlButton = styled(SubtleButton)<{ selected?: boolean }>`
|
|
|
|
width: 18px;
|
|
|
|
width: 18px;
|
|
|
|
height: 18px;
|
|
|
|
height: 18px;
|
|
|
|
background-color: ${(props) =>
|
|
|
|
background-color: ${(props) =>
|
|
|
|
props.selected ? props.theme.colors.bg0 : props.theme.colors.textMain};
|
|
|
|
props.selected ? props.theme.colors.bg0 : props.theme.colors.textMain};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
&:hover {
|
|
|
|
&:hover {
|
|
|
|
color: ${(props) => (props.selected ? props.theme.colors.bg3 : props.theme.colors.textHeadings)};
|
|
|
|
color: ${(props) => (props.selected ? props.theme.colors.bg3 : props.theme.colors.textHeadings)};
|
|
|
|
> div {
|
|
|
|
> div {
|
|
|
|
background-color: ${(props) =>
|
|
|
|
background-color: ${(props) =>
|
|
|
|
props.selected ? props.theme.colors.bg0 : props.theme.colors.textHeadings};
|
|
|
|
props.selected ? props.theme.colors.bg0 : props.theme.colors.textHeadings};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|