mirror of
				https://github.com/lingble/twenty.git
				synced 2025-10-31 04:37:56 +00:00 
			
		
		
		
	Implement new UI
This commit is contained in:
		| @@ -52,6 +52,10 @@ Once this is completed you should have: | |||||||
| - server available on: http://localhost:3000/health | - server available on: http://localhost:3000/health | ||||||
| - postgres: available on http://localhost:5432 that should contain `twenty` database | - 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 | ### 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 | 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 | ||||||
|   | |||||||
| @@ -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"> | <style type="text/css"> | ||||||
| body { | body { | ||||||
|   margin: 0; |   margin: 0; | ||||||
|   font-family: 'Source Sans Pro', sans-serif; |   font-family: 'Inter', sans-serif; | ||||||
|   -webkit-font-smoothing: antialiased; |   -webkit-font-smoothing: antialiased; | ||||||
|   -moz-osx-font-smoothing: grayscale; |   -moz-osx-font-smoothing: grayscale; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								front/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								front/package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -9,7 +9,7 @@ | |||||||
|       "version": "0.1.0", |       "version": "0.1.0", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@apollo/client": "^3.7.5", |         "@apollo/client": "^3.7.5", | ||||||
|         "@emotion/react": "^11.10.5", |         "@emotion/react": "^11.10.6", | ||||||
|         "@emotion/styled": "^11.10.5", |         "@emotion/styled": "^11.10.5", | ||||||
|         "@fortawesome/fontawesome-svg-core": "^6.2.1", |         "@fortawesome/fontawesome-svg-core": "^6.2.1", | ||||||
|         "@fortawesome/free-regular-svg-icons": "^6.2.1", |         "@fortawesome/free-regular-svg-icons": "^6.2.1", | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ | |||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@apollo/client": "^3.7.5", |     "@apollo/client": "^3.7.5", | ||||||
|     "@emotion/react": "^11.10.5", |     "@emotion/react": "^11.10.6", | ||||||
|     "@emotion/styled": "^11.10.5", |     "@emotion/styled": "^11.10.5", | ||||||
|     "@fortawesome/fontawesome-svg-core": "^6.2.1", |     "@fortawesome/fontawesome-svg-core": "^6.2.1", | ||||||
|     "@fortawesome/free-regular-svg-icons": "^6.2.1", |     "@fortawesome/free-regular-svg-icons": "^6.2.1", | ||||||
| @@ -49,7 +49,11 @@ | |||||||
|   "overrides": { |   "overrides": { | ||||||
|     "react-refresh": "0.14.0" |     "react-refresh": "0.14.0" | ||||||
|   }, |   }, | ||||||
|   "jest": {}, |   "jest": { | ||||||
|  |     "coveragePathIgnorePatterns" : [ | ||||||
|  |       ".stories.tsx$" | ||||||
|  |     ] | ||||||
|  |   }, | ||||||
|   "browserslist": { |   "browserslist": { | ||||||
|     "production": [ |     "production": [ | ||||||
|       ">0.2%", |       ">0.2%", | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ | |||||||
|     <link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" /> |     <link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" /> | ||||||
|  |  | ||||||
|     <link rel="manifest" href="%PUBLIC_URL%/manifest.json" /> |     <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> |     <title>Twenty</title> | ||||||
|   </head> |   </head> | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| body { | body { | ||||||
|   margin: 0; |   margin: 0; | ||||||
|   font-family: 'Source Sans Pro', sans-serif; |   font-family: 'Inter'; | ||||||
|   -webkit-font-smoothing: antialiased; |   -webkit-font-smoothing: antialiased; | ||||||
|   -moz-osx-font-smoothing: grayscale; |   -moz-osx-font-smoothing: grayscale; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import { | |||||||
|   createHttpLink, |   createHttpLink, | ||||||
| } from '@apollo/client'; | } from '@apollo/client'; | ||||||
| import { setContext } from '@apollo/client/link/context'; | 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 }); | const httpLink = createHttpLink({ uri: process.env.REACT_APP_API_URL }); | ||||||
|  |  | ||||||
| @@ -34,3 +36,8 @@ root.render( | |||||||
|     </BrowserRouter> |     </BrowserRouter> | ||||||
|   </ApolloProvider>, |   </ApolloProvider>, | ||||||
| ); | ); | ||||||
|  |  | ||||||
|  | declare module '@emotion/react' { | ||||||
|  |   // eslint-disable-next-line @typescript-eslint/no-empty-interface | ||||||
|  |   export interface Theme extends ThemeType {} | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								front/src/interfaces/workspace.interface.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								front/src/interfaces/workspace.interface.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | export interface Workspace { | ||||||
|  |   id: number; | ||||||
|  |   name: string; | ||||||
|  |   logo: string; | ||||||
|  | } | ||||||
| @@ -1,24 +1,37 @@ | |||||||
| import Navbar from './navbar/Navbar'; | import Navbar from './navbar/Navbar'; | ||||||
| import styled from '@emotion/styled'; | import styled from '@emotion/styled'; | ||||||
|  | import { ThemeProvider } from '@emotion/react'; | ||||||
| import { User } from '../interfaces/user.interface'; | import { User } from '../interfaces/user.interface'; | ||||||
|  | import { Workspace } from '../interfaces/workspace.interface'; | ||||||
|  | import { lightTheme } from './styles/themes'; | ||||||
|  |  | ||||||
| const StyledLayout = styled.div` | const StyledLayout = styled.div` | ||||||
|   display: flex; |   display: flex; | ||||||
|   flex-direction: column; |   flex-direction: row; | ||||||
|  |   width: 100vw; | ||||||
|   height: 100vh; |   height: 100vh; | ||||||
| `; | `; | ||||||
|  |  | ||||||
|  | const StyledRightContainer = styled.div` | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: row; | ||||||
|  |   flex: 1; | ||||||
|  | `; | ||||||
|  |  | ||||||
| type OwnProps = { | type OwnProps = { | ||||||
|   children: JSX.Element; |   children: JSX.Element; | ||||||
|   user?: User; |   user?: User; | ||||||
|  |   workspace?: Workspace; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| function AppLayout({ children, user }: OwnProps) { | function AppLayout({ children, user, workspace }: OwnProps) { | ||||||
|   return ( |   return ( | ||||||
|     <StyledLayout> |     <ThemeProvider theme={lightTheme}> | ||||||
|       <Navbar user={user} /> |       <StyledLayout> | ||||||
|       <div>{children}</div> |         <Navbar user={user} workspace={workspace} /> | ||||||
|     </StyledLayout> |         <StyledRightContainer>{children}</StyledRightContainer> | ||||||
|  |       </StyledLayout> | ||||||
|  |     </ThemeProvider> | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								front/src/layout/__stories__/AppLayout.stories.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								front/src/layout/__stories__/AppLayout.stories.tsx
									
									
									
									
									
										Normal 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> | ||||||
|  | ); | ||||||
							
								
								
									
										10
									
								
								front/src/layout/__tests__/AppLayout.test.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								front/src/layout/__tests__/AppLayout.test.tsx
									
									
									
									
									
										Normal 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'); | ||||||
|  | }); | ||||||
| @@ -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; |  | ||||||
							
								
								
									
										45
									
								
								front/src/layout/containers/WithTopBarContainer.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								front/src/layout/containers/WithTopBarContainer.tsx
									
									
									
									
									
										Normal 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; | ||||||
| @@ -1,40 +1,44 @@ | |||||||
| import styled from '@emotion/styled'; | import styled from '@emotion/styled'; | ||||||
| import { useNavigate } from 'react-router-dom'; | import { useNavigate } from 'react-router-dom'; | ||||||
|  | import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; | ||||||
|  | import { IconProp } from '@fortawesome/fontawesome-svg-core'; | ||||||
|  |  | ||||||
| type OwnProps = { | type OwnProps = { | ||||||
|   label: string; |   label: string; | ||||||
|   to: string; |   to: string; | ||||||
|   active?: boolean; |   active?: boolean; | ||||||
|  |   icon: IconProp; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| type StyledItemProps = { | type StyledItemProps = { | ||||||
|   active?: boolean; |   active?: boolean; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const StyledItem = styled.button` | const StyledItem = styled.button<StyledItemProps>` | ||||||
|   display: flex; |   display: flex; | ||||||
|   height: 60px; |   border: none; | ||||||
|   background: inherit; |   font-size: 12px; | ||||||
|   align-items: center; |  | ||||||
|   padding-left: 10px; |  | ||||||
|   padding-right: 10px; |  | ||||||
|   margin-left: 10px; |  | ||||||
|   margin-right: 10px; |  | ||||||
|   font-size: 16px; |  | ||||||
|   margin-bottom: -2px; |  | ||||||
|   cursor: pointer; |   cursor: pointer; | ||||||
|   color: ${(props: StyledItemProps) => (props.active ? 'black' : '#2e3138')}; |   background: ${(props) => (props.active ? 'rgba(0, 0, 0, 0.04)' : 'inherit')}; | ||||||
|   font-weight: ${(props: StyledItemProps) => |   padding-top: 4px; | ||||||
|     props.active ? 'bold' : 'inherit'}; |   padding-bottom: 4px; | ||||||
|   border: 0; |   padding-left: 4px; | ||||||
|   border-bottom: ${(props: StyledItemProps) => |   font-family: 'Inter'; | ||||||
|     props.active ? '2px solid black' : '2px solid #eaecee'}; |   color: ${(props) => | ||||||
|   &:hover { |     props.active ? props.theme.text100 : props.theme.text60}; | ||||||
|     border-bottom: 2px solid #2e3138; |   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(); |   const navigate = useNavigate(); | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
| @@ -45,7 +49,8 @@ function NavItem({ label, to, active }: OwnProps) { | |||||||
|       active={active} |       active={active} | ||||||
|       aria-selected={active} |       aria-selected={active} | ||||||
|     > |     > | ||||||
|       {label} |       <FontAwesomeIcon icon={icon} /> | ||||||
|  |       <StyledItemLabel>{label}</StyledItemLabel> | ||||||
|     </StyledItem> |     </StyledItem> | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								front/src/layout/navbar/NavTitle.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								front/src/layout/navbar/NavTitle.tsx
									
									
									
									
									
										Normal 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; | ||||||
| @@ -1,64 +1,62 @@ | |||||||
| import styled from '@emotion/styled'; | import styled from '@emotion/styled'; | ||||||
| import { useMatch, useResolvedPath } from 'react-router-dom'; | import { useMatch, useResolvedPath } from 'react-router-dom'; | ||||||
| import { User } from '../../interfaces/user.interface'; | import { User } from '../../interfaces/user.interface'; | ||||||
|  | import { Workspace } from '../../interfaces/workspace.interface'; | ||||||
| import NavItem from './NavItem'; | 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` | const NavbarContainer = styled.div` | ||||||
|   display: flex; |   display: flex; | ||||||
|   flex-direction: row; |   flex-direction: column; | ||||||
|   justify-content: space-between; |   background: ${(props) => props.theme.noisyBackground}; | ||||||
|   padding-left: 12px; |   min-width: 220px; | ||||||
|   height: 58px; |   padding: 8px; | ||||||
|   border-bottom: 2px solid #eaecee; |  | ||||||
| `; | `; | ||||||
|  |  | ||||||
| const NavItemsContainer = styled.div` | const NavItemsContainer = styled.div` | ||||||
|   display: flex; |   display: flex; | ||||||
|   flex-direction: row; |   flex-direction: column; | ||||||
|  |   margin-top: 40px; | ||||||
| `; | `; | ||||||
|  |  | ||||||
| type OwnProps = { | type OwnProps = { | ||||||
|   user?: User; |   user?: User; | ||||||
|  |   workspace?: Workspace; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| function Navbar({ user }: OwnProps) { | function Navbar({ workspace }: OwnProps) { | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <NavbarContainer> |       <NavbarContainer> | ||||||
|  |         {workspace && <WorkspaceContainer workspace={workspace} />} | ||||||
|         <NavItemsContainer> |         <NavItemsContainer> | ||||||
|  |           <NavTitle label="Workspace" /> | ||||||
|           <NavItem |           <NavItem | ||||||
|             label="Inbox" |             label="People" | ||||||
|             to="/" |             to="/people" | ||||||
|  |             icon={faUser} | ||||||
|             active={ |             active={ | ||||||
|               !!useMatch({ |               !!useMatch({ | ||||||
|                 path: useResolvedPath('/').pathname, |                 path: useResolvedPath('/people').pathname, | ||||||
|                 end: true, |                 end: true, | ||||||
|               }) |               }) | ||||||
|             } |             } | ||||||
|           /> |           /> | ||||||
|           <NavItem |           <NavItem | ||||||
|             label="Contacts" |             label="Companies" | ||||||
|             to="/contacts" |             to="/companies" | ||||||
|  |             icon={faBuilding} | ||||||
|             active={ |             active={ | ||||||
|               !!useMatch({ |               !!useMatch({ | ||||||
|                 path: useResolvedPath('/contacts').pathname, |                 path: useResolvedPath('/companies').pathname, | ||||||
|                 end: true, |  | ||||||
|               }) |  | ||||||
|             } |  | ||||||
|           /> |  | ||||||
|           <NavItem |  | ||||||
|             label="Insights" |  | ||||||
|             to="/insights" |  | ||||||
|             active={ |  | ||||||
|               !!useMatch({ |  | ||||||
|                 path: useResolvedPath('/insights').pathname, |  | ||||||
|                 end: true, |                 end: true, | ||||||
|               }) |               }) | ||||||
|             } |             } | ||||||
|           /> |           /> | ||||||
|         </NavItemsContainer> |         </NavItemsContainer> | ||||||
|         <ProfileContainer user={user} /> |  | ||||||
|       </NavbarContainer> |       </NavbarContainer> | ||||||
|     </> |     </> | ||||||
|   ); |   ); | ||||||
|   | |||||||
| @@ -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; |  | ||||||
							
								
								
									
										51
									
								
								front/src/layout/navbar/WorkspaceContainer.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								front/src/layout/navbar/WorkspaceContainer.tsx
									
									
									
									
									
										Normal 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; | ||||||
| @@ -1,6 +1,9 @@ | |||||||
| import { MemoryRouter } from 'react-router-dom'; | 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 NavItem from '../../../layout/navbar/NavItem'; | ||||||
|  | import { lightTheme } from '../../styles/themes'; | ||||||
|  |  | ||||||
| export default { | export default { | ||||||
|   title: 'NavItem', |   title: 'NavItem', | ||||||
| @@ -8,13 +11,17 @@ export default { | |||||||
| }; | }; | ||||||
|  |  | ||||||
| export const NavItemDefault = () => ( | export const NavItemDefault = () => ( | ||||||
|   <MemoryRouter> |   <ThemeProvider theme={lightTheme}> | ||||||
|     <NavItem label="Test" to="/test" /> |     <MemoryRouter> | ||||||
|   </MemoryRouter> |       <NavItem label="Test" to="/test" icon={faUser} /> | ||||||
|  |     </MemoryRouter> | ||||||
|  |   </ThemeProvider> | ||||||
| ); | ); | ||||||
|  |  | ||||||
| export const NavItemActive = () => ( | export const NavItemActive = () => ( | ||||||
|   <MemoryRouter initialEntries={['/test']}> |   <ThemeProvider theme={lightTheme}> | ||||||
|     <NavItem label="Test" to="/test" active={true} /> |     <MemoryRouter initialEntries={['/test']}> | ||||||
|   </MemoryRouter> |       <NavItem label="Test" to="/test" active={true} icon={faUser} /> | ||||||
|  |     </MemoryRouter> | ||||||
|  |   </ThemeProvider> | ||||||
| ); | ); | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,16 +1,16 @@ | |||||||
| import { render } from '@testing-library/react'; | import { render } from '@testing-library/react'; | ||||||
|  |  | ||||||
| import { NavbarOnInsights } from '../__stories__/Navbar.stories'; | import { NavbarOnCompanies } from '../__stories__/Navbar.stories'; | ||||||
|  |  | ||||||
| it('Checks the NavItem renders', () => { | 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', |     'aria-selected', | ||||||
|     'true', |     'true', | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   expect(getByRole('button', { name: 'Inbox' })).toHaveAttribute( |   expect(getByRole('button', { name: 'People' })).toHaveAttribute( | ||||||
|     'aria-selected', |     'aria-selected', | ||||||
|     'false', |     'false', | ||||||
|   ); |   ); | ||||||
|   | |||||||
							
								
								
									
										55
									
								
								front/src/layout/styles/themes.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								front/src/layout/styles/themes.ts
									
									
									
									
									
										Normal 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; | ||||||
							
								
								
									
										38
									
								
								front/src/layout/top-bar/TopBar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								front/src/layout/top-bar/TopBar.tsx
									
									
									
									
									
										Normal 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; | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| function Contacts() { |  | ||||||
|   return ( |  | ||||||
|     <div> |  | ||||||
|       <h1>This is the history page</h1> |  | ||||||
|     </div> |  | ||||||
|   ); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default Contacts; |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| function Insights() { |  | ||||||
|   return ( |  | ||||||
|     <div> |  | ||||||
|       <h1>This is the insights page</h1> |  | ||||||
|     </div> |  | ||||||
|   ); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default Insights; |  | ||||||
							
								
								
									
										12
									
								
								front/src/pages/companies/Companies.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								front/src/pages/companies/Companies.tsx
									
									
									
									
									
										Normal 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; | ||||||
							
								
								
									
										18
									
								
								front/src/pages/companies/__stories__/Companies.stories.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								front/src/pages/companies/__stories__/Companies.stories.tsx
									
									
									
									
									
										Normal 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> | ||||||
|  | ); | ||||||
							
								
								
									
										10
									
								
								front/src/pages/companies/__tests__/Companies.test.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								front/src/pages/companies/__tests__/Companies.test.tsx
									
									
									
									
									
										Normal 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'); | ||||||
|  | }); | ||||||
| @@ -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; |  | ||||||
| @@ -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> |  | ||||||
| ); |  | ||||||
| @@ -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'); |  | ||||||
| }); |  | ||||||
| @@ -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: |  | ||||||
|               'I’m looking for my order but couldn’t find it. Could you help me find it. I don’t know where to look for.', |  | ||||||
|           }} |  | ||||||
|         /> |  | ||||||
|         <Message |  | ||||||
|           message={{ |  | ||||||
|             id: 2, |  | ||||||
|             time: 'just now', |  | ||||||
|             user: 'Support', |  | ||||||
|             channel: 'sms', |  | ||||||
|             message: 'Hello I’m here bla bla bla', |  | ||||||
|             agent: 'Leslie A', |  | ||||||
|           }} |  | ||||||
|         /> |  | ||||||
|         <Note |  | ||||||
|           note={{ |  | ||||||
|             id: 1, |  | ||||||
|             time: 'just now', |  | ||||||
|             agent: 'LeslieA', |  | ||||||
|             message: 'Hello I’m 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; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import DiscussionPanel from '../DiscussionPanel'; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|   title: 'DiscussionPanel', |  | ||||||
|   component: DiscussionPanel, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const DiscussionPanelDefault = () => <DiscussionPanel />; |  | ||||||
| @@ -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(); |  | ||||||
| }); |  | ||||||
| @@ -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; |  | ||||||
| @@ -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; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import Composer from '../Composer'; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|   title: 'Composer', |  | ||||||
|   component: Composer, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const ComposerDefault = () => <Composer />; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import ComposerSwitch from '../ComposerSwitch'; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|   title: 'Composer', |  | ||||||
|   component: ComposerSwitch, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const ComposerSwithDefault = () => <ComposerSwitch />; |  | ||||||
| @@ -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'); |  | ||||||
| }); |  | ||||||
| @@ -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'); |  | ||||||
| }); |  | ||||||
| @@ -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; |  | ||||||
| @@ -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; |  | ||||||
| @@ -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; |  | ||||||
| @@ -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', |  | ||||||
|     }} |  | ||||||
|   /> |  | ||||||
| ); |  | ||||||
| @@ -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: |  | ||||||
|         'I’m looking for my order but couldn’t find it. Could you help me find it. I don’t know where to look for.', |  | ||||||
|     }} |  | ||||||
|   /> |  | ||||||
| ); |  | ||||||
| @@ -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 I’m here bla bla bla', |  | ||||||
|     }} |  | ||||||
|   /> |  | ||||||
| ); |  | ||||||
| @@ -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(); |  | ||||||
| }); |  | ||||||
| @@ -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(); |  | ||||||
| }); |  | ||||||
| @@ -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 I’m here bla bla bla'); |  | ||||||
|   expect(text).toBeDefined(); |  | ||||||
| }); |  | ||||||
| @@ -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: |  | ||||||
|         'I’m looking for my order but couldn’t find it. Could you help me find it. I don’t 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; |  | ||||||
| @@ -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; |  | ||||||
| @@ -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; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import ListPanel from '../ListPanel'; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|   title: 'ListPanel', |  | ||||||
|   component: ListPanel, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const ListPanelDefault = () => <ListPanel />; |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| import { MemoryRouter } from 'react-router-dom'; |  | ||||||
| import ListPanelHeader from '../ListPanelHeader'; |  | ||||||
|  |  | ||||||
| export default { |  | ||||||
|   title: 'ListPanel', |  | ||||||
|   component: ListPanelHeader, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const ListPanelHeaderDefault = () => <ListPanelHeader />; |  | ||||||
| @@ -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: |  | ||||||
|         'I’m looking for my order but couldn’t find it. Could you help me find it. I don’t know where ...', |  | ||||||
|     }} |  | ||||||
|   /> |  | ||||||
| ); |  | ||||||
| @@ -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'); |  | ||||||
| }); |  | ||||||
| @@ -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(); |  | ||||||
| }); |  | ||||||
| @@ -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(); |  | ||||||
| }); |  | ||||||
| @@ -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; |  | ||||||
| @@ -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; |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| function PluginHistory() { |  | ||||||
|   return <div></div>; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default PluginHistory; |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| import styled from '@emotion/styled'; |  | ||||||
|  |  | ||||||
| function UserActivity() { |  | ||||||
|   return; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default UserActivity; |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| import styled from '@emotion/styled'; |  | ||||||
|  |  | ||||||
| function UserInformation() { |  | ||||||
|   return; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default UserInformation; |  | ||||||
							
								
								
									
										12
									
								
								front/src/pages/people/People.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								front/src/pages/people/People.tsx
									
									
									
									
									
										Normal 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; | ||||||
							
								
								
									
										17
									
								
								front/src/pages/people/__stories__/People.stories.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								front/src/pages/people/__stories__/People.stories.tsx
									
									
									
									
									
										Normal 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> | ||||||
|  | ); | ||||||
							
								
								
									
										10
									
								
								front/src/pages/people/__tests__/People.test.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								front/src/pages/people/__tests__/People.test.tsx
									
									
									
									
									
										Normal 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'); | ||||||
|  | }); | ||||||
| @@ -8,10 +8,10 @@ sh: ## | |||||||
| 	@docker-compose exec twenty sh | 	@docker-compose exec twenty sh | ||||||
|  |  | ||||||
| front-test: ##  | front-test: ##  | ||||||
| 	@docker-compose exec twenty sh -c "npm run test" | 	@docker-compose exec twenty sh -c "cd front && npm run test" | ||||||
|  |  | ||||||
| front-coverage: ##  | front-coverage: ##  | ||||||
| 	@docker-compose exec twenty sh -c "npm run coverage" | 	@docker-compose exec twenty sh -c "cd front && npm run coverage" | ||||||
|  |  | ||||||
| front-storybook: ##  | front-storybook: ##  | ||||||
| 	@docker-compose exec twenty sh -c "npm run storybook" | 	@docker-compose exec twenty sh -c "cd front && npm run storybook" | ||||||
|   | |||||||
| @@ -1,5 +1,9 @@ | |||||||
| FROM node:18-alpine as app | FROM node:18-alpine as app | ||||||
|  |  | ||||||
|  | RUN apk update && apk upgrade && \ | ||||||
|  |     apk add --no-cache bash git openssh && \ | ||||||
|  |     apk add libc6-compat | ||||||
|  |  | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| COPY ../.. . | COPY ../.. . | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Charles Bochet
					Charles Bochet