Merge pull request #27 from twentyhq/cbo-frontend-crm-start

Implement new UI
This commit is contained in:
Charles Bochet 2023-04-10 18:06:52 +02:00 committed by GitHub
commit 693ed5746d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
69 changed files with 473 additions and 1121 deletions

View File

@ -52,6 +52,10 @@ Once this is completed you should have:
- server available on: http://localhost:3000/health
- postgres: available on http://localhost:5432 that should contain `twenty` database
### Step 3: IDE setup
If you are using VSCode, please use the `Dev Containers` extension to open the project in a container. This will allow you to run Visual Studio on top of the docker container. This will allow you to run the project without having to install node on your machine.
### Note
If you are using Docker install, make sure to ssh in the docker container during development to execute commands. You can also use `Makefile` to help you

View File

@ -1,8 +1,8 @@
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Source+Sans+Pro">
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Inter">
<style type="text/css">
body {
margin: 0;
font-family: 'Source Sans Pro', sans-serif;
font-family: 'Inter', sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}

View File

@ -9,7 +9,7 @@
"version": "0.1.0",
"dependencies": {
"@apollo/client": "^3.7.5",
"@emotion/react": "^11.10.5",
"@emotion/react": "^11.10.6",
"@emotion/styled": "^11.10.5",
"@fortawesome/fontawesome-svg-core": "^6.2.1",
"@fortawesome/free-regular-svg-icons": "^6.2.1",

View File

@ -4,7 +4,7 @@
"private": true,
"dependencies": {
"@apollo/client": "^3.7.5",
"@emotion/react": "^11.10.5",
"@emotion/react": "^11.10.6",
"@emotion/styled": "^11.10.5",
"@fortawesome/fontawesome-svg-core": "^6.2.1",
"@fortawesome/free-regular-svg-icons": "^6.2.1",
@ -49,7 +49,11 @@
"overrides": {
"react-refresh": "0.14.0"
},
"jest": {},
"jest": {
"coveragePathIgnorePatterns" : [
".stories.tsx$"
]
},
"browserslist": {
"production": [
">0.2%",

View File

@ -12,7 +12,7 @@
<link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" />
<link rel="manifest" href="%PUBLIC_URL%/manifest.json" />
<link href='https://fonts.googleapis.com/css?family=Source+Sans+Pro' rel='stylesheet' type='text/css'>
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600&display=swap" rel="stylesheet">
<title>Twenty</title>
</head>

File diff suppressed because one or more lines are too long

View File

@ -1,6 +1,6 @@
body {
margin: 0;
font-family: 'Source Sans Pro', sans-serif;
font-family: 'Inter';
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}

View File

@ -10,6 +10,8 @@ import {
createHttpLink,
} from '@apollo/client';
import { setContext } from '@apollo/client/link/context';
import '@emotion/react';
import { ThemeType } from './layout/styles/themes';
const httpLink = createHttpLink({ uri: process.env.REACT_APP_API_URL });
@ -34,3 +36,8 @@ root.render(
</BrowserRouter>
</ApolloProvider>,
);
declare module '@emotion/react' {
// eslint-disable-next-line @typescript-eslint/no-empty-interface
export interface Theme extends ThemeType {}
}

View File

@ -0,0 +1,5 @@
export interface Workspace {
id: number;
name: string;
logo: string;
}

View File

@ -1,24 +1,37 @@
import Navbar from './navbar/Navbar';
import styled from '@emotion/styled';
import { ThemeProvider } from '@emotion/react';
import { User } from '../interfaces/user.interface';
import { Workspace } from '../interfaces/workspace.interface';
import { lightTheme } from './styles/themes';
const StyledLayout = styled.div`
display: flex;
flex-direction: column;
flex-direction: row;
width: 100vw;
height: 100vh;
`;
const StyledRightContainer = styled.div`
display: flex;
flex-direction: row;
flex: 1;
`;
type OwnProps = {
children: JSX.Element;
user?: User;
workspace?: Workspace;
};
function AppLayout({ children, user }: OwnProps) {
function AppLayout({ children, user, workspace }: OwnProps) {
return (
<StyledLayout>
<Navbar user={user} />
<div>{children}</div>
</StyledLayout>
<ThemeProvider theme={lightTheme}>
<StyledLayout>
<Navbar user={user} workspace={workspace} />
<StyledRightContainer>{children}</StyledRightContainer>
</StyledLayout>
</ThemeProvider>
);
}

View File

@ -0,0 +1,19 @@
import { MemoryRouter } from 'react-router-dom';
import AppLayout from '../AppLayout';
import { ThemeProvider } from '@emotion/react';
import { lightTheme } from '../styles/themes';
export default {
title: 'AppLayout',
component: AppLayout,
};
export const AppLayoutDefault = () => (
<ThemeProvider theme={lightTheme}>
<MemoryRouter>
<AppLayout>
<div data-testid="content">Test</div>
</AppLayout>
</MemoryRouter>
</ThemeProvider>
);

View File

@ -0,0 +1,10 @@
import { render } from '@testing-library/react';
import { AppLayoutDefault } from '../__stories__/AppLayout.stories';
it('Checks the AppLayout render', () => {
const { getByTestId } = render(<AppLayoutDefault />);
const title = getByTestId('content');
expect(title).toHaveTextContent('Test');
});

View File

@ -1,16 +0,0 @@
import styled from '@emotion/styled';
type OwnProps = {
children: JSX.Element;
};
const StyledContainer = styled.div`
display: flex;
height: calc(100vh - 60px);
`;
function FullWidthContainer({ children }: OwnProps) {
return <StyledContainer>{children}</StyledContainer>;
}
export default FullWidthContainer;

View File

@ -0,0 +1,45 @@
import styled from '@emotion/styled';
import TopBar from '../top-bar/TopBar';
import { IconProp } from '@fortawesome/fontawesome-svg-core';
type OwnProps = {
children: JSX.Element;
title: string;
icon: IconProp;
};
const StyledContainer = styled.div`
display: flex;
flex: 1;
flex-direction: column;
`;
const ContentContainer = styled.div`
display: flex;
flex-direction: column;
background: ${(props) => props.theme.noisyBackground};
flex: 1;
padding-right: 12px;
padding-bottom: 12px;
`;
const ContentSubContainer = styled.div`
display: flex;
background: ${(props) => props.theme.primaryBackground};
border-radius: 8px;
border: 1px solid ${(props) => props.theme.primaryBorder};
flex: 1;
`;
function FullWidthContainer({ children, title, icon }: OwnProps) {
return (
<StyledContainer>
<TopBar title={title} icon={icon} />
<ContentContainer>
<ContentSubContainer>{children}</ContentSubContainer>
</ContentContainer>
</StyledContainer>
);
}
export default FullWidthContainer;

View File

@ -1,40 +1,44 @@
import styled from '@emotion/styled';
import { useNavigate } from 'react-router-dom';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
import { IconProp } from '@fortawesome/fontawesome-svg-core';
type OwnProps = {
label: string;
to: string;
active?: boolean;
icon: IconProp;
};
type StyledItemProps = {
active?: boolean;
};
const StyledItem = styled.button`
const StyledItem = styled.button<StyledItemProps>`
display: flex;
height: 60px;
background: inherit;
align-items: center;
padding-left: 10px;
padding-right: 10px;
margin-left: 10px;
margin-right: 10px;
font-size: 16px;
margin-bottom: -2px;
border: none;
font-size: 12px;
cursor: pointer;
color: ${(props: StyledItemProps) => (props.active ? 'black' : '#2e3138')};
font-weight: ${(props: StyledItemProps) =>
props.active ? 'bold' : 'inherit'};
border: 0;
border-bottom: ${(props: StyledItemProps) =>
props.active ? '2px solid black' : '2px solid #eaecee'};
&:hover {
border-bottom: 2px solid #2e3138;
background: ${(props) => (props.active ? 'rgba(0, 0, 0, 0.04)' : 'inherit')};
padding-top: 4px;
padding-bottom: 4px;
padding-left: 4px;
font-family: 'Inter';
color: ${(props) =>
props.active ? props.theme.text100 : props.theme.text60};
border-radius: 4px;
:hover {
background: rgba(0, 0, 0, 0.04);
color: ${(props) => props.theme.text100};
}
`;
function NavItem({ label, to, active }: OwnProps) {
const StyledItemLabel = styled.div`
display: flex;
margin-left: 8px;
`;
function NavItem({ label, icon, to, active }: OwnProps) {
const navigate = useNavigate();
return (
@ -45,7 +49,8 @@ function NavItem({ label, to, active }: OwnProps) {
active={active}
aria-selected={active}
>
{label}
<FontAwesomeIcon icon={icon} />
<StyledItemLabel>{label}</StyledItemLabel>
</StyledItem>
);
}

View File

@ -0,0 +1,22 @@
import styled from '@emotion/styled';
type OwnProps = {
label: string;
};
const StyledTitle = styled.div`
display: flex;
text-transform: uppercase;
color: ${(props) => props.theme.text30};
font-size: 12px;
font-weight: 600;
padding-top: 4px;
padding-bottom: 4px;
padding-left: 4px;
`;
function NavTitle({ label }: OwnProps) {
return <StyledTitle>{label}</StyledTitle>;
}
export default NavTitle;

View File

@ -1,64 +1,62 @@
import styled from '@emotion/styled';
import { useMatch, useResolvedPath } from 'react-router-dom';
import { User } from '../../interfaces/user.interface';
import { Workspace } from '../../interfaces/workspace.interface';
import NavItem from './NavItem';
import ProfileContainer from './ProfileContainer';
import NavTitle from './NavTitle';
import WorkspaceContainer from './WorkspaceContainer';
import { faUser } from '@fortawesome/free-regular-svg-icons';
import { faBuilding } from '@fortawesome/free-regular-svg-icons';
const NavbarContainer = styled.div`
display: flex;
flex-direction: row;
justify-content: space-between;
padding-left: 12px;
height: 58px;
border-bottom: 2px solid #eaecee;
flex-direction: column;
background: ${(props) => props.theme.noisyBackground};
min-width: 220px;
padding: 8px;
`;
const NavItemsContainer = styled.div`
display: flex;
flex-direction: row;
flex-direction: column;
margin-top: 40px;
`;
type OwnProps = {
user?: User;
workspace?: Workspace;
};
function Navbar({ user }: OwnProps) {
function Navbar({ workspace }: OwnProps) {
return (
<>
<NavbarContainer>
{workspace && <WorkspaceContainer workspace={workspace} />}
<NavItemsContainer>
<NavTitle label="Workspace" />
<NavItem
label="Inbox"
to="/"
label="People"
to="/people"
icon={faUser}
active={
!!useMatch({
path: useResolvedPath('/').pathname,
path: useResolvedPath('/people').pathname,
end: true,
})
}
/>
<NavItem
label="Contacts"
to="/contacts"
label="Companies"
to="/companies"
icon={faBuilding}
active={
!!useMatch({
path: useResolvedPath('/contacts').pathname,
end: true,
})
}
/>
<NavItem
label="Insights"
to="/insights"
active={
!!useMatch({
path: useResolvedPath('/insights').pathname,
path: useResolvedPath('/companies').pathname,
end: true,
})
}
/>
</NavItemsContainer>
<ProfileContainer user={user} />
</NavbarContainer>
</>
);

View File

@ -1,63 +0,0 @@
import styled from '@emotion/styled';
import { User } from '../../interfaces/user.interface';
type OwnProps = {
user?: User;
};
const StyledContainer = styled.button`
display: flex;
height: 60px;
background: inherit;
align-items: center;
padding-left: 10px;
padding-right: 10px;
margin-left: 10px;
margin-right: 10px;
font-size: 14px;
margin-bottom: -2px;
cursor: pointer;
border: 0;
`;
const StyledInfoContainer = styled.div`
display: flex;
flex-direction: column;
`;
const StyledEmail = styled.div`
display: flex;
`;
const StyledAvatar = styled.div`
display: flex;
width: 40px;
height: 40px;
border-radius: 40px;
background: black;
font-size: 20px;
color: white;
align-items: center;
justify-content: center;
font-weight: bold;
margin-right: 16px;
flex-shrink: 0;
`;
function ProfileContainer({ user }: OwnProps) {
return (
<StyledContainer>
<StyledAvatar>
{user?.first_name
.split(' ')
.map((n) => n[0])
.join('')}
</StyledAvatar>
<StyledInfoContainer>
<StyledEmail>{user?.email}</StyledEmail>
</StyledInfoContainer>
</StyledContainer>
);
}
export default ProfileContainer;

View File

@ -0,0 +1,51 @@
import styled from '@emotion/styled';
import { Workspace } from '../../interfaces/workspace.interface';
type OwnProps = {
workspace: Workspace;
};
const StyledContainer = styled.button`
display: inline-flex;
width: min-content;
height: 34px;
align-items: center;
cursor: pointer;
border: 0;
background: inherit;
border: 1px solid ${(props) => props.theme.primaryBorder};
border-radius: 4px;
padding: 8px;
margin-left: 4px;
`;
type StyledLogoProps = {
logo: string;
};
const StyledLogo = styled.div<StyledLogoProps>`
background: url(${(props) => props.logo});
background-size: cover;
width: 16px;
height: 16px;
border-radius: 2px;
`;
const StyledName = styled.div`
margin-left: 4px;
font-family: 'Inter';
font-weight: 500;
font-size: 14px;
font-color: ${(props) => props.theme.text0};
`;
function ProfileContainer({ workspace }: OwnProps) {
return (
<StyledContainer>
<StyledLogo logo={workspace.logo}></StyledLogo>
<StyledName>{workspace?.name}</StyledName>
</StyledContainer>
);
}
export default ProfileContainer;

View File

@ -1,6 +1,9 @@
import { MemoryRouter } from 'react-router-dom';
import { faUser } from '@fortawesome/free-regular-svg-icons';
import { ThemeProvider } from '@emotion/react';
import NavItem from '../../../layout/navbar/NavItem';
import { lightTheme } from '../../styles/themes';
export default {
title: 'NavItem',
@ -8,13 +11,17 @@ export default {
};
export const NavItemDefault = () => (
<MemoryRouter>
<NavItem label="Test" to="/test" />
</MemoryRouter>
<ThemeProvider theme={lightTheme}>
<MemoryRouter>
<NavItem label="Test" to="/test" icon={faUser} />
</MemoryRouter>
</ThemeProvider>
);
export const NavItemActive = () => (
<MemoryRouter initialEntries={['/test']}>
<NavItem label="Test" to="/test" active={true} />
</MemoryRouter>
<ThemeProvider theme={lightTheme}>
<MemoryRouter initialEntries={['/test']}>
<NavItem label="Test" to="/test" active={true} icon={faUser} />
</MemoryRouter>
</ThemeProvider>
);

File diff suppressed because one or more lines are too long

View File

@ -1,16 +1,16 @@
import { render } from '@testing-library/react';
import { NavbarOnInsights } from '../__stories__/Navbar.stories';
import { NavbarOnCompanies } from '../__stories__/Navbar.stories';
it('Checks the NavItem renders', () => {
const { getByRole } = render(<NavbarOnInsights />);
const { getByRole } = render(<NavbarOnCompanies />);
expect(getByRole('button', { name: 'Insights' })).toHaveAttribute(
expect(getByRole('button', { name: 'Companies' })).toHaveAttribute(
'aria-selected',
'true',
);
expect(getByRole('button', { name: 'Inbox' })).toHaveAttribute(
expect(getByRole('button', { name: 'People' })).toHaveAttribute(
'aria-selected',
'false',
);

View File

@ -0,0 +1,55 @@
export const lightTheme = {
noisyBackground:
'#fcfcfc url();',
primaryBackground: '#fff',
secondaryBackground: '#fcfcfc',
tertiaryBackground: '#f5f5f5',
pinkBackground: '#ffe5f4',
greenBackground: '#e6fff2',
purpleBackground: '#e0e0ff',
yellowBackground: '#fff2e7',
primaryBorder: 'rgba(0, 0, 0, 0.08)',
text100: '#000',
text80: '#333',
text60: '#666',
text40: '#999',
text30: '#b3b3b3',
text20: '#ccc',
text0: '#fff',
blue: '#1961ed',
pink: '#cc0078',
green: '#1e7e50',
purple: '#1111b7',
yellow: '#cc660a',
};
export const darkTheme = {
noisyBackground:
'#191919 url();',
primaryBackground: '#141414',
secondaryBackground: '#171717',
tertiaryBackground: '#333333',
pinkBackground: '#cc0078',
greenBackground: '#1e7e50',
purpleBackground: '#1111b7',
yellowBackground: '#cc660a',
text100: '#ffffff',
text80: '#ccc',
text60: '#999',
text40: '#666',
text30: '#4c4c4c',
text20: '#333',
text0: '#000',
blue: '#6895ec',
pink: '#ffe5f4',
green: '#e6fff2',
purple: '#e0e0ff',
yellow: '#fff2e7',
};
export type ThemeType = typeof lightTheme;

View File

@ -0,0 +1,38 @@
import styled from '@emotion/styled';
import { IconProp } from '@fortawesome/fontawesome-svg-core';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
const TopBarContainer = styled.div`
display: flex;
flex-direction: row;
height: 40px;
align-items: center;
background: ${(props) => props.theme.noisyBackground};
padding: 8px;
font-size: 14px;
color: ${(props) => props.theme.text80};
`;
const TitleContainer = styled.div`
font-family: 'Inter';
margin-left: 4px;
font-size: 14px;
`;
type OwnProps = {
title: string;
icon: IconProp;
};
function TopBar({ title, icon }: OwnProps) {
return (
<>
<TopBarContainer>
<FontAwesomeIcon icon={icon} />
<TitleContainer data-testid="top-bar-title">{title}</TitleContainer>
</TopBarContainer>
</>
);
}
export default TopBar;

View File

@ -1,9 +0,0 @@
function Contacts() {
return (
<div>
<h1>This is the history page</h1>
</div>
);
}
export default Contacts;

View File

@ -1,9 +0,0 @@
function Insights() {
return (
<div>
<h1>This is the insights page</h1>
</div>
);
}
export default Insights;

View File

@ -0,0 +1,12 @@
import { faBuilding } from '@fortawesome/free-regular-svg-icons';
import WithTopBarContainer from '../../layout/containers/WithTopBarContainer';
function Companies() {
return (
<WithTopBarContainer title="Companies" icon={faBuilding}>
<></>
</WithTopBarContainer>
);
}
export default Companies;

View File

@ -0,0 +1,18 @@
import { MemoryRouter } from 'react-router-dom';
import Companies from '../Companies';
import { ThemeProvider } from '@emotion/react';
import { lightTheme } from '../../../layout/styles/themes';
import AppLayout from '../../../layout/AppLayout';
export default {
title: 'Companies',
component: Companies,
};
export const CompaniesDefault = () => (
<ThemeProvider theme={lightTheme}>
<MemoryRouter>
<Companies />
</MemoryRouter>
</ThemeProvider>
);

View File

@ -0,0 +1,10 @@
import { render } from '@testing-library/react';
import { CompaniesDefault } from '../__stories__/Companies.stories';
it('Checks the Companies page render', () => {
const { getByTestId } = render(<CompaniesDefault />);
const title = getByTestId('top-bar-title');
expect(title).toHaveTextContent('Companies');
});

View File

@ -1,18 +0,0 @@
import FullWidthContainer from '../../layout/containers/FullWidthContainer';
import DiscussionPanel from './discussion-panel/DiscussionPanel';
import ListPanel from './list-panel/ListPanel';
import PluginPanel from './plugin-panel/PluginPanel';
function Inbox() {
return (
<FullWidthContainer>
<>
<ListPanel />
<DiscussionPanel />
<PluginPanel />
</>
</FullWidthContainer>
);
}
export default Inbox;

View File

@ -1,13 +0,0 @@
import { MemoryRouter } from 'react-router-dom';
import Inbox from '../Inbox';
export default {
title: 'Inbox',
component: Inbox,
};
export const InboxDefault = () => (
<MemoryRouter>
<Inbox />
</MemoryRouter>
);

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { InboxDefault } from '../__stories__/Inbox.stories';
it('Checks the Inbox page render', () => {
const { getAllByRole } = render(<InboxDefault />);
const button = getAllByRole('button');
expect(button[0]).toHaveTextContent('Sylvie Vartan');
});

View File

@ -1,100 +0,0 @@
import styled from '@emotion/styled';
import Composer from './composer/Composer';
import Booking, { BookingEvent } from './events/Booking';
import Message, { MessageEvent } from './events/Message';
import Note, { NoteEvent } from './events/Note';
export type Event = BookingEvent | MessageEvent | NoteEvent;
const StyledPanel = styled.div`
display: flex;
flex-grow: 1;
flex-direction: column;
`;
const EventsContainer = styled.div`
display: flex;
flex-grow: 1;
flex-direction: column;
padding: 32px;
`;
const StyledToday = styled.div`
font-size: 12px;
color: #2e3138;
border-bottom: 1px solid #eaecee;
margin-top: 32px;
padding-bottom: 8px;
margin-bottom: 8px;
`;
const ComposerContainer = styled.div`
display: flex;
padding: 32px;
flex-direction: column;
flex-grow: 1;
`;
function DiscussionPanel() {
return (
<StyledPanel>
<EventsContainer>
<Booking
booking={{
id: 1,
time: 'Wed, Sep 10, 2022',
user: 'Georges',
nights: 4,
guests: 5,
price: '756.90$',
listing: 'Rochefort Montagne',
dateRange: 'Mon, Sep 30 - Fri, Oct 2',
}}
/>
<StyledToday>Today</StyledToday>
<Message
message={{
id: 1,
time: '2 hours ago',
user: 'Georges Alain',
channel: 'sms',
message:
'Im looking for my order but couldnt find it. Could you help me find it. I dont know where to look for.',
}}
/>
<Message
message={{
id: 2,
time: 'just now',
user: 'Support',
channel: 'sms',
message: 'Hello Im here bla bla bla',
agent: 'Leslie A',
}}
/>
<Note
note={{
id: 1,
time: 'just now',
agent: 'LeslieA',
message: 'Hello Im here bla bla bla',
}}
/>
<Message
message={{
id: 3,
time: 'just now',
user: 'Georges Alain',
channel: 'sms',
message: 'Thank you !',
}}
/>
</EventsContainer>
<ComposerContainer>
<Composer />
</ComposerContainer>
</StyledPanel>
);
}
export default DiscussionPanel;

View File

@ -1,8 +0,0 @@
import DiscussionPanel from '../DiscussionPanel';
export default {
title: 'DiscussionPanel',
component: DiscussionPanel,
};
export const DiscussionPanelDefault = () => <DiscussionPanel />;

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { DiscussionPanelDefault } from '../__stories__/DiscussionPanel.stories';
it('Checks the discussion panel render', () => {
const { getAllByText } = render(<DiscussionPanelDefault />);
const text = getAllByText('Rochefort Montagne');
expect(text).toBeDefined();
});

View File

@ -1,57 +0,0 @@
import styled from '@emotion/styled';
import ComposerSwitch from './ComposerSwitch';
const StyledContainer = styled.div`
display: flex;
flex-direction: column;
border: 2px solid #000000;
border-radius: 12px;
`;
const StyledInputContainer = styled.div`
display: flex;
padding: 8px;
justify-content: center;
& > textarea {
width: 95%;
border: none;
outline: none;
font-family: 'Source Sans Pro';
&::placeholder {
font-family: 'Source Sans Pro';
}
}
`;
const ActionContainer = styled.div`
display: flex;
padding: 16px;
justify-content: flex-end;
`;
const PrimaryButton = styled.button`
background: black;
font-weight: bold;
color: white;
padding: 16px 24px;
border: 0;
border-radius: 12px;
cursor: pointer;
`;
function Composer() {
return (
<StyledContainer>
<ComposerSwitch />
<StyledInputContainer>
<textarea rows={5} placeholder="Type to chat..."></textarea>
</StyledInputContainer>
<ActionContainer>
<PrimaryButton>Reply by SMS</PrimaryButton>
</ActionContainer>
</StyledContainer>
);
}
export default Composer;

View File

@ -1,49 +0,0 @@
import styled from '@emotion/styled';
const StyledContainer = styled.div`
display: flex;
flex-direction: row;
border-bottom: 2px solid #eaecee;
padding: 0px 12px;
`;
const SwitchTab = styled.button`
display: flex;
border-bottom: 2px solid #eaecee;
margin-bottom: -2px;
padding: 12px;
cursor: pointer;
color: #2e3138;
background: inherit;
border: 0;
&:hover {
border-bottom: 2px solid #2e3138;
}
`;
const SwitchTabActive = styled.button`
display: flex;
border: 0;
border-bottom: 2px solid black;
margin-bottom: -2px;
padding: 12px;
cursor: pointer;
color: black;
font-weight: bold;
background: inherit;
`;
function ComposerSwitch() {
return (
<StyledContainer>
<SwitchTabActive>Reply</SwitchTabActive>
<SwitchTab>Call</SwitchTab>
<SwitchTab>Note</SwitchTab>
<SwitchTab>Transfer</SwitchTab>
<SwitchTab>Help</SwitchTab>
</StyledContainer>
);
}
export default ComposerSwitch;

View File

@ -1,8 +0,0 @@
import Composer from '../Composer';
export default {
title: 'Composer',
component: Composer,
};
export const ComposerDefault = () => <Composer />;

View File

@ -1,8 +0,0 @@
import ComposerSwitch from '../ComposerSwitch';
export default {
title: 'Composer',
component: ComposerSwitch,
};
export const ComposerSwithDefault = () => <ComposerSwitch />;

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { ComposerDefault } from '../__stories__/Composer.stories';
it('Checks the composer render', () => {
const { getAllByRole } = render(<ComposerDefault />);
const button = getAllByRole('button');
expect(button[0]).toHaveTextContent('Reply');
});

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { ComposerSwithDefault } from '../__stories__/ComposerSwitch.stories';
it('Checks the composer switch render', () => {
const { getAllByRole } = render(<ComposerSwithDefault />);
const button = getAllByRole('button');
expect(button[0]).toHaveTextContent('Reply');
});

View File

@ -1,80 +0,0 @@
import styled from '@emotion/styled';
export type BookingEvent = {
id: number;
user: string;
time: string;
listing: string;
nights: number;
guests: number;
price: string;
dateRange: string;
};
type OwnProps = {
booking: BookingEvent;
};
const StyledBooking = styled.div`
display: flex;
flex-direction: column;
`;
const StyledLabel = styled.div`
font-size: 12px;
color: #2e3138;
margin-bottom: 8px;
`;
const StyledContainer = styled.div`
display: flex;
padding: 16px;
flex-direction: row;
border: 1px solid #000000;
border-radius: 12px;
`;
const StyledPicture = styled.div`
background: #2e3138;
width: 50px;
height: 42px;
margin-right: 16px;
`;
const StyledContent = styled.div`
display: flex;
flex-direction: column;
justify-content: center;
`;
const StyledListing = styled.div`
font-size: 16px;
font-weight: bold;
`;
const StyledDetails = styled.div`
font-size: 12px;
color: #2e3138;
`;
function Booking({ booking }: OwnProps) {
return (
<StyledBooking>
<StyledLabel>
{booking.time} · {booking.user} booked a trip
</StyledLabel>
<StyledContainer>
<StyledPicture />
<StyledContent>
<StyledListing>{booking.listing}</StyledListing>
<StyledDetails>
{booking.dateRange} · {booking.nights} nights · {booking.guests}{' '}
guests · {booking.price}
</StyledDetails>
</StyledContent>
</StyledContainer>
</StyledBooking>
);
}
export default Booking;

View File

@ -1,90 +0,0 @@
import styled from '@emotion/styled';
export type MessageEvent = {
id: number;
user: string;
time: string;
channel: string;
message: string;
agent?: string;
};
type OwnProps = {
message: MessageEvent;
};
const StyledMessage = styled.div`
display: flex;
margin-top: 12px;
margin-bottom: 20px;
`;
const StyledAvatar = styled.div`
display: flex;
width: 40px;
height: 40px;
border-radius: 40px;
background: black;
font-size: 20px;
color: white;
align-items: center;
justify-content: center;
font-weight: bold;
margin-right: 16px;
flex-shrink: 0;
`;
const StyledContent = styled.div``;
const StyledTitle = styled.div`
display: flex;
align-items: center;
`;
const StyledUser = styled.div`
font-size: 16px;
color: black;
font-weight: bold;
`;
const StyledTime = styled.div`
margin-left: 8px;
font-size: 12px;
color: #2e3138;
`;
const StyledAgent = styled.div`
text-decoration: underline;
margin-left: 4px;
font-size: 12px;
color: #2e3138;
`;
const StyledDetails = styled.div`
margin-top: 8px;
`;
function Message({ message }: OwnProps) {
return (
<StyledMessage>
<StyledAvatar>
{message.user
.split(' ')
.map((n) => n[0])
.join('')}
</StyledAvatar>
<StyledContent>
<StyledTitle>
<StyledUser>{message.user}</StyledUser>
<StyledTime>
{message.time} ({message.channel})
</StyledTime>
{message.agent && <StyledAgent>by {message.agent}</StyledAgent>}
</StyledTitle>
<StyledDetails>{message.message}</StyledDetails>
</StyledContent>
</StyledMessage>
);
}
export default Message;

View File

@ -1,52 +0,0 @@
import styled from '@emotion/styled';
export type NoteEvent = {
id: number;
time: string;
message: string;
agent: string;
};
type OwnProps = {
note: NoteEvent;
};
const StyledNote = styled.div`
display: flex;
background: #f8f9fa;
border-left: 4px solid black;
padding: 8px 20px;
flex-direction: column;
margin-top: 12px;
margin-bottom: 20px;
`;
const StyledLabel = styled.div`
font-size: 12px;
color: #2e3138;
margin-bottom: 8px;
display: flex;
flex-direction: row;
`;
const StyledAgent = styled.div`
text-decoration: underline;
margin-left: 4px;
font-size: 12px;
color: #2e3138;
`;
const StyledMessage = styled.div``;
function Note({ note }: OwnProps) {
return (
<StyledNote>
<StyledLabel>
Internal Note {note.time} <StyledAgent>by {note.agent}</StyledAgent>
</StyledLabel>
<StyledMessage>{note.message}</StyledMessage>
</StyledNote>
);
}
export default Note;

View File

@ -1,21 +0,0 @@
import Booking from '../Booking';
export default {
title: 'DiscussionPanel',
component: Booking,
};
export const BookingDefault = () => (
<Booking
booking={{
id: 1,
time: 'Wed, Sep 10, 2022',
user: 'Georges',
nights: 4,
guests: 5,
price: '756.90$',
listing: 'Rochefort Montagne',
dateRange: 'Mon, Sep 30 - Fri, Oct 2',
}}
/>
);

View File

@ -1,19 +0,0 @@
import Message from '../Message';
export default {
title: 'DiscussionPanel',
component: Message,
};
export const MessageDefault = () => (
<Message
message={{
id: 1,
time: '2 hours ago',
user: 'Georges Alain',
channel: 'sms',
message:
'Im looking for my order but couldnt find it. Could you help me find it. I dont know where to look for.',
}}
/>
);

View File

@ -1,17 +0,0 @@
import Note from '../Note';
export default {
title: 'DiscussionPanel',
component: Note,
};
export const NoteDefault = () => (
<Note
note={{
id: 1,
time: 'just now',
agent: 'LeslieA',
message: 'Hello Im here bla bla bla',
}}
/>
);

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { BookingDefault } from '../__stories__/Booking.stories';
it('Checks the booking event render', () => {
const { getAllByText } = render(<BookingDefault />);
const text = getAllByText('Rochefort Montagne');
expect(text).toBeDefined();
});

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { MessageDefault } from '../__stories__/Message.stories';
it('Checks the booking event render', () => {
const { getAllByText } = render(<MessageDefault />);
const text = getAllByText('Georges Alain');
expect(text).toBeDefined();
});

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { NoteDefault } from '../__stories__/Note.stories';
it('Checks the booking event render', () => {
const { getAllByText } = render(<NoteDefault />);
const text = getAllByText('Hello Im here bla bla bla');
expect(text).toBeDefined();
});

View File

@ -1,50 +0,0 @@
import styled from '@emotion/styled';
import ListPanelHeader from './ListPanelHeader';
import ListPanelItem from './ListPanelItem';
const StyledList = styled.div`
display: flex;
width: 325px;
flex-direction: column;
border-right: 2px solid #eaecee;
`;
export type Task = {
id: number;
targetUser: string;
label: string;
time: string;
lastMessage: string;
};
function ListPanel() {
const tasks: Task[] = [
{
id: 1,
targetUser: 'Sylvie Vartan',
label: 'Guest at #xxx property',
time: '3h',
lastMessage:
'Im looking for my order but couldnt find it. Could you help me find it. I dont know where ...',
},
{
id: 2,
targetUser: 'Johnny Halliday',
label: 'Guest at #xxx property',
time: '4h',
lastMessage: 'Hello, this is Johnny',
},
];
return (
<StyledList>
<>
<ListPanelHeader />
{tasks.map((item) => (
<ListPanelItem key={item.id} task={item} />
))}
</>
</StyledList>
);
}
export default ListPanel;

View File

@ -1,17 +0,0 @@
import styled from '@emotion/styled';
const StyledHeader = styled.div`
display: flex;
font-weight: bold;
align-items: center;
padding: 16px 24px;
font-size: 18px;
letter-spacing: 1px;
border-bottom: 1px solid #eaecee;
`;
function ListPanelHeader() {
return <StyledHeader>6 tasks waiting</StyledHeader>;
}
export default ListPanelHeader;

View File

@ -1,98 +0,0 @@
import styled from '@emotion/styled';
import { Task } from './ListPanel';
type OwnProps = {
task: Task;
};
const StyledListItem = styled.button`
display: flex;
padding: 16px 24px;
flex-direction: column;
color: #2e3138;
border: 0;
border-bottom: 1px solid #eaecee;
cursor: pointer;
font-family: inherit;
text-align: inherit;
align-items: inherit;
background: #f1f3f5;
`;
const StyledHeader = styled.div`
display: flex;
justify-content: space-between;
`;
const StyledAvatarAndTitle = styled.div`
display: flex;
`;
const StyledAvatar = styled.div`
display: flex;
width: 40px;
height: 40px;
border-radius: 40px;
background: #52555b;
font-size: 20px;
color: white;
align-items: center;
justify-content: center;
font-weight: bold;
`;
const StyledTitle = styled.div`
display: flex;
flex-direction: column;
margin-left: 8px;
`;
const StyledName = styled.div`
font-weight: bold;
font-size: 18px;
color: black;
`;
const StyledLabel = styled.div`
display: flex;
font-size: 14px;
`;
const StyledTime = styled.div`
display: flex;
justify-self: flex-end;
color: #7d8187;
font-size: 14px;
`;
const StyledContent = styled.div`
display: flex;
color: #52555b;
font-size: 14px;
margin-top: 8px;
`;
function ListPanelItem({ task }: OwnProps) {
return (
<StyledListItem>
<StyledHeader>
<StyledAvatarAndTitle>
<StyledAvatar>
{task.targetUser
.split(' ')
.map((n) => n[0])
.join('')}
</StyledAvatar>
<StyledTitle>
<StyledName>{task.targetUser}</StyledName>
<StyledLabel>{task.label}</StyledLabel>
</StyledTitle>
</StyledAvatarAndTitle>
<StyledTime>{task.time}</StyledTime>
</StyledHeader>
<StyledContent>{task.lastMessage} </StyledContent>
</StyledListItem>
);
}
export default ListPanelItem;

View File

@ -1,8 +0,0 @@
import ListPanel from '../ListPanel';
export default {
title: 'ListPanel',
component: ListPanel,
};
export const ListPanelDefault = () => <ListPanel />;

View File

@ -1,9 +0,0 @@
import { MemoryRouter } from 'react-router-dom';
import ListPanelHeader from '../ListPanelHeader';
export default {
title: 'ListPanel',
component: ListPanelHeader,
};
export const ListPanelHeaderDefault = () => <ListPanelHeader />;

View File

@ -1,19 +0,0 @@
import ListPanelItem from '../ListPanelItem';
export default {
title: 'ListPanel',
component: ListPanelItem,
};
export const ListPanelItemDefault = () => (
<ListPanelItem
task={{
id: 1,
targetUser: 'Sylvie Vartan',
label: 'Guest at #xxx property',
time: '3h',
lastMessage:
'Im looking for my order but couldnt find it. Could you help me find it. I dont know where ...',
}}
/>
);

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { ListPanelDefault } from '../__stories__/ListPanel.stories';
it('Checks the task list render', () => {
const { getAllByRole } = render(<ListPanelDefault />);
const button = getAllByRole('button');
expect(button[0]).toHaveTextContent('Sylvie Vartan');
});

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { ListPanelHeaderDefault } from '../__stories__/ListPanelHeader.stories';
it('Checks the ListPanelHeader render', () => {
const { getAllByText } = render(<ListPanelHeaderDefault />);
const text = getAllByText('6 tasks waiting');
expect(text).toBeDefined();
});

View File

@ -1,10 +0,0 @@
import { render } from '@testing-library/react';
import { ListPanelItemDefault } from '../__stories__/ListPanelItem.stories';
it('Checks the ListPanelItem render', () => {
const { getAllByText } = render(<ListPanelItemDefault />);
const text = getAllByText('Sylvie Vartan');
expect(text).toBeDefined();
});

View File

@ -1,27 +0,0 @@
import styled from '@emotion/styled';
import PluginPanelNav from './PluginPanelNav';
import PluginHistory from './plugin-history/PanelHistory';
const StyledPanel = styled.div`
display: flex;
width: 350px;
border-left: 1px solid #eaecee;
`;
const StyledContainer = styled.div`
display: flex;
flex-grow: 1;
`;
function PluginPanel() {
return (
<StyledPanel>
<StyledContainer>
<PluginHistory />
</StyledContainer>
<PluginPanelNav />
</StyledPanel>
);
}
export default PluginPanel;

View File

@ -1,46 +0,0 @@
import styled from '@emotion/styled';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
import { faClone } from '@fortawesome/free-regular-svg-icons';
const StyledNav = styled.div`
display: flex;
flex-direction: column;
width: 60px;
border-left: 1px solid #eaecee;
background: #f1f3f5;
`;
const StyledNavItem = styled.div`
display: flex;
width: 60px;
border-bottom: 1px solid #eaecee;
padding: 22px;
cursor: pointer;
`;
function PluginPanelNav() {
return (
<StyledNav>
<StyledNavItem>
<FontAwesomeIcon icon={faClone} size="lg" />
</StyledNavItem>
<StyledNavItem>
<FontAwesomeIcon icon={faClone} size="lg" />
</StyledNavItem>
<StyledNavItem>
<FontAwesomeIcon icon={faClone} size="lg" />
</StyledNavItem>
<StyledNavItem>
<FontAwesomeIcon icon={faClone} size="lg" />
</StyledNavItem>
<StyledNavItem>
<FontAwesomeIcon icon={faClone} size="lg" />
</StyledNavItem>
<StyledNavItem>
<FontAwesomeIcon icon={faClone} size="lg" />
</StyledNavItem>
</StyledNav>
);
}
export default PluginPanelNav;

View File

@ -1,5 +0,0 @@
function PluginHistory() {
return <div></div>;
}
export default PluginHistory;

View File

@ -1,7 +0,0 @@
import styled from '@emotion/styled';
function UserActivity() {
return;
}
export default UserActivity;

View File

@ -1,7 +0,0 @@
import styled from '@emotion/styled';
function UserInformation() {
return;
}
export default UserInformation;

View File

@ -0,0 +1,12 @@
import { faUser } from '@fortawesome/free-regular-svg-icons';
import WithTopBarContainer from '../../layout/containers/WithTopBarContainer';
function People() {
return (
<WithTopBarContainer title="People" icon={faUser}>
<></>
</WithTopBarContainer>
);
}
export default People;

View File

@ -0,0 +1,17 @@
import { MemoryRouter } from 'react-router-dom';
import People from '../People';
import { ThemeProvider } from '@emotion/react';
import { lightTheme } from '../../../layout/styles/themes';
export default {
title: 'People',
component: People,
};
export const PeopleDefault = () => (
<ThemeProvider theme={lightTheme}>
<MemoryRouter>
<People />
</MemoryRouter>
</ThemeProvider>
);

View File

@ -0,0 +1,10 @@
import { render } from '@testing-library/react';
import { PeopleDefault } from '../__stories__/People.stories';
it('Checks the People page render', () => {
const { getByTestId } = render(<PeopleDefault />);
const title = getByTestId('top-bar-title');
expect(title).toHaveTextContent('People');
});

View File

@ -8,10 +8,10 @@ sh: ##
@docker-compose exec twenty sh
front-test: ##
@docker-compose exec twenty sh -c "npm run test"
@docker-compose exec twenty sh -c "cd front && npm run test"
front-coverage: ##
@docker-compose exec twenty sh -c "npm run coverage"
@docker-compose exec twenty sh -c "cd front && npm run coverage"
front-storybook: ##
@docker-compose exec twenty sh -c "npm run storybook"
@docker-compose exec twenty sh -c "cd front && npm run storybook"

View File

@ -1,5 +1,9 @@
FROM node:18-alpine as app
RUN apk update && apk upgrade && \
apk add --no-cache bash git openssh && \
apk add libc6-compat
WORKDIR /app
COPY ../.. .