Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

OV-144: Add Script Generation For Videos #228

Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
20 commits
Select commit Hold shift + click to select a range
6d11703
OV-144: + initial helper to transofrm payload to text to AI
lfelix3011 Sep 5, 2024
62d62ce
Merge branch 'next' of https://github.com/BinaryStudioAcademy/bsa-202…
lfelix3011 Sep 5, 2024
087b3ac
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 5, 2024
82d80b9
OV-144: + generate text with AI using chat component
lfelix3011 Sep 6, 2024
c95d555
Merge branch 'next' of https://github.com/BinaryStudioAcademy/bsa-202…
lfelix3011 Sep 7, 2024
35b598e
OV-144: * reponse to be an object of title and description insted of …
lfelix3011 Sep 7, 2024
b11249e
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 9, 2024
2f9ff34
Merge branch 'next' of https://github.com/BinaryStudioAcademy/bsa-202…
lfelix3011 Sep 9, 2024
043bfc1
OV-144: * genearete text with corresponding animation and visual
lfelix3011 Sep 10, 2024
eb48f1f
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 10, 2024
94875b9
OV-42: * use general loader when waiting respond
lfelix3011 Sep 10, 2024
ca317b6
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 10, 2024
427079c
OV-144: * suggestion, to make code simplier, and tone not required
lfelix3011 Sep 11, 2024
62aa93b
Merge branch 'next' of https://github.com/BinaryStudioAcademy/bsa-202…
lfelix3011 Sep 11, 2024
3176164
OV-144: * styles that were remove on last merge
lfelix3011 Sep 11, 2024
71eab4a
Merge branch 'next' of https://github.com/BinaryStudioAcademy/bsa-202…
lfelix3011 Sep 11, 2024
809d2d8
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 11, 2024
9a43874
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 11, 2024
e5413b8
Merge branch 'task/OV-42-add-chat-component' of https://github.com/Bi…
lfelix3011 Sep 11, 2024
8677bcf
OV-144: * use route form bundle insted of relative import, and chain …
lfelix3011 Sep 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions frontend/src/bundles/common/components/components.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,10 @@ export {
Spacer,
Stack,
Tab,
TabList,
TabPanel,
TabPanels,
Tabs,
Text,
Tooltip,
VStack,
Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { MessageSender } from '~/bundles/chat/enums/enums.js';
import {
Button,
FormProvider,
Expand All @@ -6,7 +7,13 @@ import {
Textarea,
VStack,
} from '~/bundles/common/components/components.js';
import { useAppForm } from '~/bundles/common/hooks/hooks.js';
import { DataStatus } from '~/bundles/common/enums/enums.js';
import {
useAppForm,
useAppSelector,
useCallback,
useMemo,
} from '~/bundles/common/hooks/hooks.js';
import {
type GenerateVideoScriptRequestDto,
generateVideoScriptValidationSchema,
Expand All @@ -19,6 +26,12 @@ type Properties = {
};

const GenerateScriptForm: React.FC<Properties> = ({ onSubmit }) => {
const { messages, dataStatus } = useAppSelector(({ chat }) => ({
messages: chat.messages.filter(
(message) => message.sender === MessageSender.AI,
),
dataStatus: chat.dataStatus,
}));
const form = useAppForm<GenerateVideoScriptRequestDto>({
initialValues: DEFAULT_GENERATE_SCRIPT_PAYLOAD,
validationSchema: generateVideoScriptValidationSchema,
Expand All @@ -27,6 +40,18 @@ const GenerateScriptForm: React.FC<Properties> = ({ onSubmit }) => {

const { handleSubmit, values } = form;

const isTextGenerated: boolean = useMemo(() => {
return messages && messages.length > 0;
lfelix3011 marked this conversation as resolved.
Show resolved Hide resolved
}, [messages]);

const getButtonLabel = useCallback(() => {
if (dataStatus === DataStatus.PENDING) {
return 'stop';
lfelix3011 marked this conversation as resolved.
Show resolved Hide resolved
}

return isTextGenerated ? 'Re-Generate' : 'Generate script';
lfelix3011 marked this conversation as resolved.
Show resolved Hide resolved
}, [dataStatus, isTextGenerated]);

return (
<FormProvider value={form}>
<form onSubmit={handleSubmit}>
Expand Down Expand Up @@ -57,7 +82,7 @@ const GenerateScriptForm: React.FC<Properties> = ({ onSubmit }) => {
/>
<Button
type="submit"
label="Generate script"
label={getButtonLabel()}
sx={{ mt: '50px' }}
isDisabled={values.topic.trim().length === 0}
/>
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import { type ComponentWithAs, type IconProps } from '@chakra-ui/react';

import { Flex, Icon, Text } from '~/bundles/common/components/components.js';

type Properties = {
message: string;
icon?: ComponentWithAs<'svg', IconProps>;
};

const GenerateScriptPlaceholderContent: React.FC<Properties> = ({
message,
icon,
}) => (
<Flex
gap="10px"
direction="column"
justify="center"
alignItems="center"
width="100%"
height="100%"
>
{icon && (
<Icon
as={icon}
color="brand.secondary.300"
opacity="0.5"
w={6}
h={6}
/>
)}
<Text
color="gray.400"
variant="H3"
w="40%"
minWidth="175px"
textAlign="center"
fontStyle="italic"
>
{message}
</Text>
</Flex>
);

export { GenerateScriptPlaceholderContent };
Original file line number Diff line number Diff line change
@@ -1,28 +1,51 @@
import { Icon, Text, VStack } from '@chakra-ui/react';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';

import { VStack } from '~/bundles/common/components/components.js';
import { DataStatus } from '~/bundles/common/enums/data-status.enum.js';
import { useAppSelector, useMemo } from '~/bundles/common/hooks/hooks.js';
import { IconName } from '~/bundles/common/icons/icons.js';
import { type VideoScript } from '~/bundles/common/types/types.js';

import { GenerateScriptPlaceholderContent } from '../generate-script-placeholder-content/generate-script-placeholder-content.js';
import { GenerateScriptScene } from '../generate-script-scene/generate-script-scene.js';

type Properties = {
videoScripts: VideoScript[];
};

const GenerateScriptPlaceholder: React.FC<Properties> = ({ videoScripts }) => {
const { dataStatus } = useAppSelector(({ chat }) => ({
dataStatus: chat.dataStatus,
}));
const isGeneratedTextEmpty = useMemo(
() => videoScripts.length === 0,
[videoScripts],
);

const GenerateScriptPlaceholder: React.FC = () => {
return (
<VStack w="full" p="40px" gap="10px">
<Icon
as={FontAwesomeIcon}
icon={IconName.SCROLL}
color="brand.secondary.300"
opacity="0.5"
size="2x"
/>
<Text
color="gray.400"
variant="H3"
w="40%"
minWidth="175px"
textAlign="center"
fontStyle="italic"
>
Here you will see your generated script
</Text>
<VStack
w="full"
paddingX="40px"
gap="10px"
height={'100vh'}
maxH={'550px'}
overflowY={'auto'}
>
{dataStatus === DataStatus.PENDING ? (
<GenerateScriptPlaceholderContent message="Loading..." />
) : (isGeneratedTextEmpty ? (
<GenerateScriptPlaceholderContent
message="Here you will see your generated script"
icon={IconName.SCROLL}
/>
) : (
<>
{videoScripts.map((videoScript, index) => (
<GenerateScriptScene
key={index}
videoScript={videoScript}
/>
))}
</>
))}
</VStack>
);
};
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
.line-1 {
/* position: relative;
lfelix3011 marked this conversation as resolved.
Show resolved Hide resolved
top: 50%;
width: 100%; */
margin: 0 auto;
border-right: 2px solid rgba(255, 255, 255, 0.75);
/* text-align: center;
white-space: nowrap; */
/* overflow: hidden; */
/* transform: translateY(-50%); */
min-height: 470px;
width: 100%;
}

.anim-typewriter {
animation:
typewriter 4s steps(44) 1s 1 normal both,
blinkTextCursor 500ms steps(44) infinite normal;
}
@keyframes typewriter {
from {
width: 0;
}
to {
width: 100%;
}
}
@keyframes blinkTextCursor {
from {
border-right-color: rgba(255, 255, 255, 0.75);
}
to {
border-right-color: transparent;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import { Box, Heading } from '~/bundles/common/components/components.js';
import { useCallback, useState } from '~/bundles/common/hooks/hooks.js';
import { type VideoScript } from '~/bundles/common/types/types.js';

import styles from './styles.module.css';

type Properties = {
videoScript: VideoScript;
};

const GenerateScriptScene: React.FC<Properties> = ({ videoScript }) => {
const { title, description } = videoScript;
const [animationDone, setAnimationDone] = useState(false);

const handleAnimationEnd = useCallback(() => {
setAnimationDone(true);
}, [setAnimationDone]);

return (
<Box mb={'10'}>
<Heading as="h4" variant={'H4'} color={'typography.600'}>
{title}
</Heading>
<Box as="p" className={animationDone ? styles['typing-effect--done'] : styles['typing-effect']} onAnimationEnd={handleAnimationEnd}>{description}</Box>
</Box>
);
};

export { GenerateScriptScene };
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
@keyframes typing {
from {
width: 0;
}
to {
width: 100%;
}
}

@keyframes blink-caret {
from, to {
border-color: transparent;
}
50% {
border-color: black;
}
}

.typing-effect {
overflow: hidden;
border-right: 2px solid black;
width: 0;
animation: typing 3s steps(30, end) forwards, blink-caret 0.75s step-end infinite;
}
.typing-effect--done {
white-space: normal;
overflow: visible;
border-right: none;
}
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
import { type GenerateTextRequestDto } from '~/bundles/chat/chat.js';
import { MessageSender } from '~/bundles/chat/enums/message-sender.js';
import { actions as chatActions } from '~/bundles/chat/store/chat.js';
import {
Heading,
HStack,
Expand All @@ -6,23 +9,65 @@ import {
TabPanel,
TabPanels,
Tabs,
} from '@chakra-ui/react';

import { useCallback } from '~/bundles/common/hooks/hooks.js';
} from '~/bundles/common/components/components.js';
import {
useAppDispatch,
useAppSelector,
useCallback,
useMemo,
} from '~/bundles/common/hooks/hooks.js';
import { type VideoScript } from '~/bundles/common/types/video-script.type.js';
import { type GenerateVideoScriptRequestDto } from '~/bundles/video-scripts/video-scripts.js';

import {
getVideoScriptMessageFromPayload,
sanitizeJsonString,
} from '../../helpers/helpers.js';
lfelix3011 marked this conversation as resolved.
Show resolved Hide resolved
import { GenerateScriptForm } from '../generate-script-form/generate-script-form.js';
import { GenerateScriptPlaceholder } from '../generate-script-placeholder/generate-script-placeholder.js';

const GenerateScriptView: React.FC = () => {
const dispatch = useAppDispatch();
const { messages } = useAppSelector(({ chat }) => ({
messages: chat.messages.filter(
(message) => message.sender === MessageSender.AI,
),
}));

const handleGenerateVideoScriptSubmit = useCallback(
// eslint-disable-next-line @typescript-eslint/no-unused-vars
(payload: GenerateVideoScriptRequestDto): void => {
// TODO dispatch video script generation action
const sendMessageRequest: GenerateTextRequestDto = {
message: getVideoScriptMessageFromPayload(payload, messages),
};
void dispatch(chatActions.sendMessage(sendMessageRequest));
},
[],
[messages, dispatch],
);

const lastGeneratedScript: VideoScript[] = useMemo(() => {
if (!messages || messages.length === 0) {
return [];
}

const lastMessage = messages.at(-1);
if (!lastMessage) {
return [];
}

try {
const sanitizedJson = sanitizeJsonString(lastMessage.text);
const videoScripts: VideoScript[] = JSON.parse(sanitizedJson);
return videoScripts;
} catch {
return [
{
title: 'Scene',
description: lastMessage.text,
},
];
}
}, [messages]);

return (
<>
<Heading
Expand Down Expand Up @@ -56,11 +101,16 @@ const GenerateScriptView: React.FC = () => {
</TabList>
<TabPanels>
<TabPanel p={0}>
<HStack justify="space-between">
<HStack
justify="space-between"
alignItems={'flex-start'}
>
<GenerateScriptForm
onSubmit={handleGenerateVideoScriptSubmit}
/>
<GenerateScriptPlaceholder />
<GenerateScriptPlaceholder
videoScripts={lastGeneratedScript}
/>
</HStack>
</TabPanel>
</TabPanels>
Expand Down
Loading
Loading