This commit is contained in:
sajjad 2023-11-19 01:16:43 +03:30
parent deb6e1fbcd
commit 4c6b767736
20 changed files with 800 additions and 66 deletions

82
package-lock.json generated
View File

@ -21,10 +21,13 @@
"react-redux": "^8.1.3",
"react-scripts": "5.0.1",
"styled-components": "^6.1.1",
"swiper": "^11.0.4",
"web-vitals": "^2.1.4"
},
"devDependencies": {
"react-router-dom": "^6.18.0"
"react-router-dom": "^6.18.0",
"sass": "^1.69.5",
"tailwindcss": "^3.3.5"
}
},
"node_modules/@aashutoshrathi/word-wrap": {
@ -9812,6 +9815,12 @@
"url": "https://opencollective.com/immer"
}
},
"node_modules/immutable": {
"version": "4.3.4",
"resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.4.tgz",
"integrity": "sha512-fsXeu4J4i6WNWSikpI88v/PcVflZz+6kMhUfIwc5SY+poQRPnaf5V7qds6SUyUN3cVxEzuCab7QIoLOQ+DQ1wA==",
"devOptional": true
},
"node_modules/import-fresh": {
"version": "3.3.0",
"resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz",
@ -15950,6 +15959,23 @@
"resolved": "https://registry.npmjs.org/sanitize.css/-/sanitize.css-13.0.0.tgz",
"integrity": "sha512-ZRwKbh/eQ6w9vmTjkuG0Ioi3HBwPFce0O+v//ve+aOq1oeCy7jMV2qzzAlpsNuqpqCBjjriM1lbtZbF/Q8jVyA=="
},
"node_modules/sass": {
"version": "1.69.5",
"resolved": "https://registry.npmjs.org/sass/-/sass-1.69.5.tgz",
"integrity": "sha512-qg2+UCJibLr2LCVOt3OlPhr/dqVHWOa9XtZf2OjbLs/T4VPSJ00udtgJxH3neXZm+QqX8B+3cU7RaLqp1iVfcQ==",
"devOptional": true,
"dependencies": {
"chokidar": ">=3.0.0 <4.0.0",
"immutable": "^4.0.0",
"source-map-js": ">=0.6.2 <2.0.0"
},
"bin": {
"sass": "sass.js"
},
"engines": {
"node": ">=14.0.0"
}
},
"node_modules/sass-loader": {
"version": "12.6.0",
"resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-12.6.0.tgz",
@ -16979,6 +17005,24 @@
"boolbase": "~1.0.0"
}
},
"node_modules/swiper": {
"version": "11.0.4",
"resolved": "https://registry.npmjs.org/swiper/-/swiper-11.0.4.tgz",
"integrity": "sha512-qtUxILrD4aD++rpKzGrkz3IAWL92f9uTrDwjb6HaNLmPvJhZCE/83DL+9w4kIgDDJeF6QKalV47rMBN77UOVYQ==",
"funding": [
{
"type": "patreon",
"url": "https://www.patreon.com/swiperjs"
},
{
"type": "open_collective",
"url": "http://opencollective.com/swiper"
}
],
"engines": {
"node": ">= 4.7.0"
}
},
"node_modules/symbol-tree": {
"version": "3.2.4",
"resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz",
@ -17422,16 +17466,16 @@
}
},
"node_modules/typescript": {
"version": "5.2.2",
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
"integrity": "sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==",
"version": "4.9.5",
"resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz",
"integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==",
"peer": true,
"bin": {
"tsc": "bin/tsc",
"tsserver": "bin/tsserver"
},
"engines": {
"node": ">=14.17"
"node": ">=4.2.0"
}
},
"node_modules/unbox-primitive": {
@ -25549,6 +25593,12 @@
"resolved": "https://registry.npmjs.org/immer/-/immer-9.0.21.tgz",
"integrity": "sha512-bc4NBHqOqSfRW7POMkHd51LvClaeMXpm8dx0e8oE2GORbq5aRK7Bxl4FyzVLdGtLmvLKL7BTDBG5ACQm4HWjTA=="
},
"immutable": {
"version": "4.3.4",
"resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.4.tgz",
"integrity": "sha512-fsXeu4J4i6WNWSikpI88v/PcVflZz+6kMhUfIwc5SY+poQRPnaf5V7qds6SUyUN3cVxEzuCab7QIoLOQ+DQ1wA==",
"devOptional": true
},
"import-fresh": {
"version": "3.3.0",
"resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz",
@ -29777,6 +29827,17 @@
"resolved": "https://registry.npmjs.org/sanitize.css/-/sanitize.css-13.0.0.tgz",
"integrity": "sha512-ZRwKbh/eQ6w9vmTjkuG0Ioi3HBwPFce0O+v//ve+aOq1oeCy7jMV2qzzAlpsNuqpqCBjjriM1lbtZbF/Q8jVyA=="
},
"sass": {
"version": "1.69.5",
"resolved": "https://registry.npmjs.org/sass/-/sass-1.69.5.tgz",
"integrity": "sha512-qg2+UCJibLr2LCVOt3OlPhr/dqVHWOa9XtZf2OjbLs/T4VPSJ00udtgJxH3neXZm+QqX8B+3cU7RaLqp1iVfcQ==",
"devOptional": true,
"requires": {
"chokidar": ">=3.0.0 <4.0.0",
"immutable": "^4.0.0",
"source-map-js": ">=0.6.2 <2.0.0"
}
},
"sass-loader": {
"version": "12.6.0",
"resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-12.6.0.tgz",
@ -30562,6 +30623,11 @@
}
}
},
"swiper": {
"version": "11.0.4",
"resolved": "https://registry.npmjs.org/swiper/-/swiper-11.0.4.tgz",
"integrity": "sha512-qtUxILrD4aD++rpKzGrkz3IAWL92f9uTrDwjb6HaNLmPvJhZCE/83DL+9w4kIgDDJeF6QKalV47rMBN77UOVYQ=="
},
"symbol-tree": {
"version": "3.2.4",
"resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz",
@ -30888,9 +30954,9 @@
}
},
"typescript": {
"version": "5.2.2",
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
"integrity": "sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==",
"version": "4.9.5",
"resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz",
"integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==",
"peer": true
},
"unbox-primitive": {

View File

@ -16,6 +16,7 @@
"react-redux": "^8.1.3",
"react-scripts": "5.0.1",
"styled-components": "^6.1.1",
"swiper": "^11.0.4",
"web-vitals": "^2.1.4"
},
"scripts": {
@ -43,6 +44,8 @@
]
},
"devDependencies": {
"react-router-dom": "^6.18.0"
"react-router-dom": "^6.18.0",
"sass": "^1.69.5",
"tailwindcss": "^3.3.5"
}
}

View File

@ -1,4 +1,4 @@
a {
text-decoration: none;
color: inherit;
}
text-decoration: none;
color: inherit;
}

View File

@ -1,18 +1,42 @@
import './App.css';
import { Outlet } from 'react-router-dom';
import Grid from '@mui/material/Grid';
import MainNav from './components/Global/main-nav/MainNav';
import "./App.css";
import { Outlet } from "react-router-dom";
import Grid from "@mui/material/Grid";
import MainNav from "./components/Global/main-nav/MainNav";
import { createTheme, ThemeProvider } from "@mui/material/styles";
const theme = createTheme({
palette: {
primary: {
main: "#0095f6",
// light: will be calculated from palette.primary.main,
// dark: will be calculated from palette.primary.main,
// contrastText: will be calculated to contrast with palette.primary.main
},
secondary: {
main: "#E0C2FF",
light: "#F5EBFF",
// dark: will be calculated from palette.secondary.main,
contrastText: "#47008F",
},
black: {
main: "#000",
},
},
});
function App() {
return (
<div className="App">
<Grid container spacing={2}>
<Grid item xs={2}>
<MainNav />
<ThemeProvider theme={theme}>
<div className="App">
<Grid container spacing={2}>
<Grid item xs={1.8}>
<MainNav />
</Grid>
<Grid item xs={10.2}>
<Outlet />
</Grid>
</Grid>
</Grid>
<Outlet />
</div>
</div>
</ThemeProvider>
);
}

View File

@ -1,26 +1,163 @@
// base
import { Link } from "react-router-dom";
import { useState } from "react";
// components
import MainNavItem from "./MainNavItem";
import HomeIcon from '@mui/icons-material/Home';
import { Divider, List, IconButton } from "@mui/material";
import MuiDrawer from "@mui/material/Drawer";
// Icons
import ChevronLeftIcon from "@mui/icons-material/ChevronLeft";
import ChevronRightIcon from "@mui/icons-material/ChevronRight";
import HomeOutlinedIcon from "@mui/icons-material/HomeOutlined";
import HomeSharpIcon from "@mui/icons-material/HomeSharp";
import SearchOutlinedIcon from "@mui/icons-material/SearchOutlined";
import SearchSharpIcon from "@mui/icons-material/SearchSharp";
import ExploreOutlinedIcon from "@mui/icons-material/ExploreOutlined";
import ExploreSharpIcon from "@mui/icons-material/ExploreSharp";
import SendIcon from "@mui/icons-material/Send";
import SendOutlinedIcon from "@mui/icons-material/SendOutlined";
import AddBoxOutlinedIcon from "@mui/icons-material/AddBoxOutlined";
// styles
import { styled, useTheme, Theme, CSSObject } from "@mui/material/styles";
const DrawerHeader = styled("div")(({ theme }) => ({
display: "flex",
alignItems: "center",
justifyContent: "flex-end",
padding: theme.spacing(0, 1),
// necessary for content to be below app bar
...theme.mixins.toolbar,
}));
const openedMixin = (theme: Theme): CSSObject => ({
width: 240,
transition: theme.transitions.create("width", {
easing: theme.transitions.easing.sharp,
duration: theme.transitions.duration.enteringScreen,
}),
overflowX: "hidden",
});
const closedMixin = (theme: Theme): CSSObject => ({
transition: theme.transitions.create("width", {
easing: theme.transitions.easing.sharp,
duration: theme.transitions.duration.leavingScreen,
}),
overflowX: "hidden",
width: `calc(${theme.spacing(7)} + 1px)`,
[theme.breakpoints.up("sm")]: {
width: `calc(${theme.spacing(7)} + 1px)`,
},
});
const Drawer = styled(MuiDrawer, {
shouldForwardProp: (prop) => prop !== "open",
})(({ theme, open }) => ({
width: 240,
flexShrink: 0,
whiteSpace: "nowrap",
boxSizing: "border-box",
...(open && {
...openedMixin(theme),
"& .MuiDrawer-paper": openedMixin(theme),
}),
...(!open && {
...closedMixin(theme),
"& .MuiDrawer-paper": closedMixin(theme),
}),
}));
const MainNav = () => {
const navItems = [
{
text: 'Home',
path: '/',
icon: HomeIcon
}
];
return (
<div>
{
navItems.map((navItem, i) => {
return (
<Link to={navItem.path} key={i}>
<MainNavItem text={navItem.text} icon={navItem.icon} />
</Link>
)
})
}
</div >
)
const [activeItem, setActiveItem] = useState(0);
const navItems = [
{
text: "Home",
path: "/",
icon: HomeOutlinedIcon,
activeIcon: HomeSharpIcon,
},
{
text: "Search",
path: null,
icon: SearchOutlinedIcon,
activeIcon: SearchSharpIcon,
},
{
text: "Explore",
path: "/",
icon: ExploreOutlinedIcon,
activeIcon: ExploreSharpIcon,
},
{
text: "Messages",
path: "/messages",
icon: SendOutlinedIcon,
activeIcon: SendIcon,
},
{
text: "Create",
path: "/",
icon: AddBoxOutlinedIcon,
},
{
text: "Profile",
path: "/",
icon: SearchOutlinedIcon,
},
];
const [open, setOpen] = useState(true);
const handleListItemClick = (index) => {
setActiveItem(index);
if (!navItems[index].path) {
handleDrawerClose();
}
};
const handleDrawerOpen = () => {
setOpen(true);
};
const handleDrawerClose = () => {
setOpen(false);
};
return (
<div>
<Drawer variant="permanent" open={open}>
<DrawerHeader>
{open ? (
<IconButton className="p-0" onClick={handleDrawerClose}>
<ChevronLeftIcon />
</IconButton>
) : (
<IconButton className="p-0" onClick={handleDrawerOpen}>
<ChevronRightIcon />
</IconButton>
)}
</DrawerHeader>
<Divider />
<List>
{navItems.map((navItem, i) => {
return (
<Link to={navItem.path} key={i}>
<MainNavItem
onClick={(e) => {
!navItem.path ?? e.preventDefault();
handleListItemClick(i);
}}
text={navItem.text}
icon={navItem.icon}
activeIcon={navItem.activeIcon}
index={i}
activeItem={activeItem}
open={open}
/>
</Link>
);
})}
</List>
</Drawer>
</div>
);
};
export default MainNav;
export default MainNav;

View File

@ -1,5 +1,14 @@
.main__nav__item {
.main__nav__item--text span {
font-size: 20px;
}
.MainNavItem {
.MainNavItemText span {
color: gray;
font-size: 16px;
}
.MainNavItemIcon {
color: black;
}
}
.MainNavItemActive {
.MainNavItemText span {
color: black;
}
}

View File

@ -1,16 +1,44 @@
import { ListItemButton, ListItemText, ListItemIcon } from '@mui/material';
import styles from './MainNav.module.scss';
import { ListItemButton, ListItemText, ListItemIcon } from "@mui/material";
import styled from "./MainNav.module.scss";
const MainNavItem = ({ text, path, icon }) => {
const MainNavItemIcon = icon;
return (
<ListItemButton className={styles.main__nav__item} href={path}>
<ListItemIcon className="main__nav__item--icon" sx={{ 'min-width': '30px' }}>
<MainNavItemIcon fontSize="large" />
</ListItemIcon>
<ListItemText className="main__nav__item--text" primary={text} />
</ListItemButton>
);
const MainNavItem = ({
text,
path,
icon,
index,
onClick,
activeItem,
activeIcon,
open,
}) => {
const MainNavItemIcon = index === activeItem ? activeIcon : icon;
return (
<ListItemButton
onClick={onClick}
className={`${styled.MainNavItem} ${
index === activeItem ? styled.MainNavItemActive : ""
}`}
href={path}
sx={{
minHeight: 48,
justifyContent: open ? "initial" : "center",
px: 2.5,
}}
>
<ListItemIcon
className={styled.MainNavItemIcon}
sx={{
minWidth: open ? 0 : "60px",
mr: 1,
justifyContent: 'center'
}}
>
<MainNavItemIcon fontSize="medium" />
</ListItemIcon>
<ListItemText className={styled.MainNavItemText} primary={text} />
</ListItemButton>
);
};
export default MainNavItem;
export default MainNavItem;

View File

@ -0,0 +1,13 @@
import styled from "./ProfileImage.module.scss";
const ProfileImage = ({ image, title, active }) => {
return (
<div
className={`${styled.ImageListItem} ${
active ? styled.ImageListItemActive : ""
}`}
>
<img src={image} alt={title ? title : "image"} loading="lazy" />
</div>
);
};
export default ProfileImage;

View File

@ -0,0 +1,18 @@
.ImageListItem {
position: relative;
width: 100%;
height: 100%;
overflow: hidden;
margin: 0 auto;
border-radius: 50%;
&.ImageListItemActive {
background: linear-gradient(to right, red, orange) padding-box,
linear-gradient(to right, red, orange) border-box;
border: 2px solid transparent;
}
img {
width: 100%;
height: 100%;
object-fit: cover;
}
}

View File

@ -0,0 +1,108 @@
import styled from "./Slider.module.scss";
import React from "react";
import ArrowCircleRightTwoToneIcon from "@mui/icons-material/ArrowCircleRightTwoTone";
import ArrowCircleLeftTwoToneIcon from "@mui/icons-material/ArrowCircleLeftTwoTone";
import { useRef } from "react";
// Import Swiper React components
import { Swiper, SwiperSlide } from "swiper/react";
// Import Swiper styles
import "swiper/css";
import "swiper/css/pagination";
import "swiper/css/navigation";
// import required modules
import { Navigation } from "swiper/modules";
const Slider = ({ body, items, sliderBodyCallBack, slidesPerView = 8 }) => {
const swiperRef = useRef();
return (
<Swiper
slidesPerView={slidesPerView}
spaceBetween={3}
modules={[Navigation]}
className={styled.mySwiper}
onBeforeInit={(swiper) => {
swiperRef.current = swiper;
}}
>
<div
className={styled.CustomSwiperPrevButton}
onClick={() => swiperRef.current?.slidePrev()}
>
{items && items.length > 1 && (
<ArrowCircleLeftTwoToneIcon fontSize="medium" />
)}
</div>
{items &&
items.map((item, i) => (
<SwiperSlide key={i}>
{sliderBodyCallBack({ data: item, index: i })}
</SwiperSlide>
))}
<div
className={styled.CustomSwiperNextButton}
onClick={() => swiperRef.current?.slideNext()}
>
{items && items.length > 1 && (
<ArrowCircleRightTwoToneIcon fontSize="medium" />
)}
</div>
</Swiper>
);
};
export default Slider;
// function Body() {
// return null;
// }
// class Slider extends React.Component {
// static Body = Body;
// constructor(props) {
// super(props);
// this.swiperRef = React.createRef(null);
// }
// render() {
// // const swiperRef = useRef();
// const { children, items, sendDataToParent, handleCallBack } = this.props;
// const body = children;
// return (
// <Swiper
// slidesPerView={8}
// spaceBetween={3}
// modules={[Navigation]}
// className={styled.mySwiper}
// onBeforeInit={(swiper) => {
// this.swiperRef.current = swiper;
// }}
// >
// <div
// className={styled.CustomSwiperPrevButton}
// onClick={() => this.swiperRef.current?.slidePrev()}
// >
// <ArrowCircleLeftTwoToneIcon fontSize="medium" />
// </div>
// {items &&
// items.map((item, i) => {
// sendDataToParent(item, i);
// console.log(body);
// return (
// <SwiperSlide key={i}>
// {body ? body.props.children : null}
// {handleCallBack(item)}
// </SwiperSlide>
// );
// })}
// <div
// className={styled.CustomSwiperNextButton}
// onClick={() => this.swiperRef.current?.slideNext()}
// >
// <ArrowCircleRightTwoToneIcon fontSize="medium" />
// </div>
// </Swiper>
// );
// }
// }
// export default Slider;

View File

@ -0,0 +1,25 @@
.ImageListItemBar {
text-align: center;
div {
font-size: 12px;
}
}
.mySwiper {
position: relative;
float: right;
width: 100%;
}
.CustomSwiperPrevButton {
position: absolute;
bottom: 45%;
left: 15px;
z-index: 99;
cursor: pointer;
}
.CustomSwiperNextButton {
position: absolute;
bottom: 45%;
right: 15px;
z-index: 99;
cursor: pointer;
}

View File

@ -0,0 +1,26 @@
// components
import ProfileImage from "../profile-image/ProfileImage";
import { Button } from "@mui/material";
// styles
import styled from "./UserCard.module.scss";
const UserCard = ({ title, image, subTitle }) => {
return (
<div className="flex items-center justify-between">
<div className="flex items-center">
<div className={styled.UserCardImage}>
<ProfileImage image={image} title="user profile image" />
</div>
<div className="ml-3">
<div className="text-[14px] font-semibold">sajjad_talkhabi_</div>
<div className={`${styled.UserCardSubTitle} text-sm/4 text-gray-500`}>
followed by sajjad_talkhabi_
</div>
</div>
</div>
<Button variant="text" className="primary capitalize">
Follow
</Button>
</div>
);
};
export default UserCard;

View File

@ -0,0 +1,11 @@
.UserCardImage {
width: 2.5vw;
height: 2.5vw;
}
.UserCardSubTitle {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
max-width: 160px;
}

View File

@ -0,0 +1,117 @@
// styles
import styled from "./Item.module.scss";
// components
import Slider from "../Global/slider/Slider";
import { TextField, Button } from "@mui/material";
import ProfileImage from "../Global/profile-image/ProfileImage";
// icons
import MoreHorizOutlinedIcon from "@mui/icons-material/MoreHorizOutlined";
import FavoriteBorderIcon from "@mui/icons-material/FavoriteBorder";
import FavoriteSharpIcon from "@mui/icons-material/FavoriteSharp";
import ChatBubbleOutlineOutlinedIcon from "@mui/icons-material/ChatBubbleOutlineOutlined";
import SendSharpIcon from "@mui/icons-material/SendSharp";
import BookmarkBorderOutlinedIcon from "@mui/icons-material/BookmarkBorderOutlined";
import BookmarkSharpIcon from "@mui/icons-material/BookmarkSharp";
// colors
import { red } from "@mui/material/colors";
const Post = () => {
const posts = [
{
img: "",
},
];
return (
<div className={styled.PostItem}>
<div
className={`${styled.PostItemHeader} flex items-center justify-between`}
>
<div className={`${styled.PostItemHeaderAccount} flex items-center`}>
<div className={styled.PostItemHeaderAccountImage}>
<ProfileImage image={posts[0].img} alt={"fad"} active={true} />
</div>
<div
className={`${styled.PostItemHeaderAccountContent} flex ml-3 text-sm `}
>
<div className="font-semibold">sajjad_talkhabi_</div>
<div className={styled.PostItemHeaderAccountDot}>
<span></span>
</div>
<div className="text-gray-500">1h</div>
<div className="flex items-center">
<div className={styled.PostItemHeaderAccountDot}>
<span></span>
</div>
<div className="text-sky-600 capitalize tracking-wide">
follow
</div>
</div>
</div>
</div>
<MoreHorizOutlinedIcon />
</div>
<div className={styled.PostItemBody}>
<div className="flex">
<Slider
slidesPerView={1}
items={posts}
sliderBodyCallBack={({ data }) => (
<img
src={data.img}
alt="alt"
className="object-fit w-full rounded mt-3"
/>
)}
/>
</div>
<div className="flex items-center justify-between mt-2">
<div className="flex items-center">
<FavoriteBorderIcon sx={{ fontSize: 30 }} />
{/* <FavoriteSharpIcon sx={{ color: red[600], fontSize: 30 }} /> */}
<ChatBubbleOutlineOutlinedIcon
sx={{ fontSize: 30 }}
className="mx-2"
/>
<SendSharpIcon
sx={{
transform: "rotate(-25deg)",
marginTop: "-5px",
fontSize: 30,
}}
/>
</div>
<div>
<BookmarkBorderOutlinedIcon sx={{ fontSize: 30 }} />
<BookmarkSharpIcon sx={{ fontSize: 30 }} />
</div>
</div>
<div className="block text-sm font-semibold mt-4">6 likes</div>
<div className="block text-sm font-semibold">
sajjad_talkhabi_
<span className="font-normal ml-3">تجربش کردی؟🥲💔</span>
</div>
<div className="flex items-center mb-2">
<div className="block text-[14px] font-semibold mt-3">
mohammadh.safa
<span className="font-normal ml-3">اره جیب خالی😂</span>
</div>
</div>
<div className="flex items-center">
<TextField
placeholder="Add a comment"
color="black"
multiline
rows={1}
fullWidth
maxRows={4}
variant="standard"
/>
<Button variant="text">Post</Button>
</div>
</div>
</div>
);
};
export default Post;

View File

@ -0,0 +1,20 @@
.PostItem {
.PostItemHeader {
.PostItemHeaderAccount {
.PostItemHeaderAccountImage {
height: 2.5vw;
width: 2.5vw;
}
}
}
.PostItemHeaderAccountDot {
span {
height: 4px;
width: 4px;
margin: 0 8px;
background-color: #bbb;
border-radius: 50%;
display: inline-block;
}
}
}

View File

@ -1,3 +1,6 @@
@tailwind base;
@tailwind components;
@tailwind utilities;
body {
margin: 0;
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',

View File

@ -1,7 +1,97 @@
const Dashboard = () => (
// components
import Slider from "../../components/Global/slider/Slider";
import PostItem from "../../components/posts/Item";
import ProfileImage from "../../components/Global/profile-image/ProfileImage";
import { ImageListItemBar, ImageListItem, Grid, Divider } from "@mui/material";
// styles
import styled from "./Dashboard.module.scss";
import UserCard from "../../components/Global/user-card/UserCard";
const Dashboard = () => {
const stories = [
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
{
img: "",
author: "sajjad",
title: "story item",
},
];
return (
<div className="App">
dashboard
<Grid container spacing={2}>
<Grid item xs={8} paddingLeft={5} className="h-full">
<div className="mt-3 flex ml-60">
<Slider
items={stories}
sliderBodyCallBack={({ data }) => <StoryItem story={data} />}
/>
</div>
<div className="ml-60 mr-20 px-20">
<PostItem />
<div className="mt-2">
<Divider />
</div>
</div>
</Grid>
<Grid item xs={4} className="px-20">
<div className="mt-7">
<UserCard image={stories[0].img} />
</div>
</Grid>
</Grid>
</div>
);
);
};
const StoryItem = ({ story }) => {
return (
<ImageListItem>
<div className={styled.StoryItemImage}>
<ProfileImage image={story.img} tilte={story.title} active={true} />
</div>
<ImageListItemBar
className={styled.ImageListItemBar}
position="below"
title={story.author}
/>
</ImageListItem>
);
};
export default Dashboard;

View File

@ -0,0 +1,27 @@
.ImageListItem {
position: relative;
width: 4vw;
height: 4vw;
overflow: hidden;
border-radius: 50%;
img {
width: 100%;
height: 100%;
object-fit: cover;
}
}
.StoryItemImage {
width: 4vw;
height: 4vw;
}
.swiper-button-next {
background-image: url("");
background-repeat: no-repeat;
background-size: 100% auto;
background-position: center;
}
.swiper-button-next::after {
display: none;
}

9
tailwind.config.js Normal file
View File

@ -0,0 +1,9 @@
/** @type {import('tailwindcss').Config} */
module.exports = {
important: true,
content: ["./src/**/*.{js,jsx,ts,tsx}"],
theme: {
extend: {},
},
plugins: [],
};