How to put different content for each cards - reactjs

I am using material ui album template for my react app
I want to put different image and different text for each cards for this when I remove this array it create the problems in responsiveness of the cards
I tried to put separte grid for each cards but thats some how solve the issue but that responsiveness of the template does not remain same
here is my demo code
https://codesandbox.io/s/material-demo-pz8df

Make sure you're updating the right cards array. Use an array of objects as well, where each object has a key for an image-link and one for the description. In the .map() we'll use these values to render the content. Here's a working sandbox: https://codesandbox.io/s/material-demo-3v44c
The responsiveness will work like expected.
Working code:
import React from "react";
import AppBar from "#material-ui/core/AppBar";
import Button from "#material-ui/core/Button";
import CameraIcon from "#material-ui/icons/PhotoCamera";
import Card from "#material-ui/core/Card";
import CardActions from "#material-ui/core/CardActions";
import CardContent from "#material-ui/core/CardContent";
import CardMedia from "#material-ui/core/CardMedia";
import CssBaseline from "#material-ui/core/CssBaseline";
import Grid from "#material-ui/core/Grid";
import Toolbar from "#material-ui/core/Toolbar";
import Typography from "#material-ui/core/Typography";
import { makeStyles } from "#material-ui/core/styles";
import Container from "#material-ui/core/Container";
import Link from "#material-ui/core/Link";
function MadeWithLove() {
return (
<Typography variant="body2" color="textSecondary" align="center">
{"Built with love by the "}
<Link color="inherit" href="https://material-ui.com/">
Material-UI
</Link>
{" team."}
</Typography>
);
}
const useStyles = makeStyles(theme => ({
icon: {
marginRight: theme.spacing(2)
},
heroContent: {
backgroundColor: theme.palette.background.paper,
padding: theme.spacing(8, 0, 6)
},
heroButtons: {
marginTop: theme.spacing(4)
},
cardGrid: {
paddingTop: theme.spacing(8),
paddingBottom: theme.spacing(8)
},
card: {
height: "100%",
display: "flex",
flexDirection: "column"
},
cardMedia: {
paddingTop: "56.25%" // 16:9
},
cardContent: {
flexGrow: 1
},
footer: {
backgroundColor: theme.palette.background.paper,
padding: theme.spacing(6)
}
}));
const cards = [
{
img:
"https://images.unsplash.com/photo-1564135624576-c5c88640f235?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=crop&w=3300&q=80",
desc: "Campsite"
},
{
img:
"https://images.unsplash.com/photo-1564198879220-63f2734f7cec?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=crop&w=2072&q=80",
desc: "Space"
}
];
export default function Album() {
const classes = useStyles();
return (
<React.Fragment>
<CssBaseline />
<AppBar position="relative">
<Toolbar>
<CameraIcon className={classes.icon} />
<Typography variant="h6" color="inherit" noWrap>
Album layout
</Typography>
</Toolbar>
</AppBar>
<main>
{/* Hero unit */}
<div className={classes.heroContent}>
<Container maxWidth="sm">
<Typography
component="h1"
variant="h2"
align="center"
color="textPrimary"
gutterBottom
>
Album layout
</Typography>
<Typography
variant="h5"
align="center"
color="textSecondary"
paragraph
>
Something short and leading about the collection below—its
contents, the creator, etc. Make it short and sweet, but not too
short so folks don&apos;t simply skip over it entirely.
</Typography>
<div className={classes.heroButtons}>
<Grid container spacing={2} justify="center">
<Grid item>
<Button variant="contained" color="primary">
Main call to action
</Button>
</Grid>
<Grid item>
<Button variant="outlined" color="primary">
Secondary action
</Button>
</Grid>
</Grid>
</div>
</Container>
</div>
<Container className={classes.cardGrid} maxWidth="md">
{/* End hero unit */}
<Grid container spacing={4}>
{cards.map(card => (
<Grid item key={card} xs={12} sm={6} md={4}>
<Card className={classes.card}>
<CardMedia
className={classes.cardMedia}
image={card.img}
title="Image title"
/>
<CardContent className={classes.cardContent}>
<Typography gutterBottom variant="h5" component="h2">
Heading
</Typography>
<Typography>{card.desc}</Typography>
</CardContent>
<CardActions>
<Button size="small" color="primary">
View
</Button>
<Button size="small" color="primary">
Edit
</Button>
</CardActions>
</Card>
</Grid>
))}
</Grid>
</Container>
</main>
{/* Footer */}
<footer className={classes.footer}>
<Typography variant="h6" align="center" gutterBottom>
Footer
</Typography>
<Typography
variant="subtitle1"
align="center"
color="textSecondary"
component="p"
>
Something here to give the footer a purpose!
</Typography>
<MadeWithLove />
</footer>
{/* End footer */}
</React.Fragment>
);
}

Related

trying to display Cards horizontally 3 items per row materia ui

I am pretty new to Material Ui and ReactJs. i am trying to display cards horizontally and each row has 3 cards. but now its currently displaying only 1 column.
this is a file that calls the cards
import * as React from 'react';
import Jobs from './Jobs'
const jobs1=[{
company:"razer",
jobtitle:"Data engineer",
},{
company:"Huawei",
jobtitle:"Data Engineer"
},{
company:"Huawei",
jobtitle:"Data Engineer"
}]
export default function Header() {
return (
jobs1.map((jobs1)=>(
<Jobs jobs1={jobs1}/>
)))
}
this is the cards file
import React from 'react';
import Card from '#mui/material/Card';
import CardContent from '#mui/material/CardContent';
import CardMedia from '#mui/material/CardMedia';
import img from './image/Razer.png'
import Typography from '#mui/material/Typography';
import {Grid,Box } from '#mui/material'
const Jobs = ({jobs1: {company, jobtitle}})=>{
return (
<Card sx={{ maxWidth: 300 }}>
<CardMedia
component="img"
height="100"
sx={{ width: 100}}
image={img}
alt="razer"
/>
<CardContent>
<Typography style={{color: '#bdbdbd'}}gutterBottom variant="h9" component="div">
{company}
</Typography>
<Typography component="div">
<Box sx={{ fontWeight: 'bold' }}>
{jobtitle}
</Box>
</Typography>
<Grid container direction="row" alignItems="center" gap={14}>
<Typography sx={{fontStyle: 'italic'}} style={{color: '#bdbdbd'}}inline variant="body1" align="left">
1 minute ago
</Typography>
<Typography inline variant="body1" >
<Box sx={{ borderRadius: '12px', p: 1,opacity:"0.5", border: '0.5px solid',color: "#64dd17",bgcolor:"#ccff90" }}>Data</Box>
</Typography>
</Grid>
<Typography style={{color: '#bdbdbd'}}>____________________________</Typography>
</CardContent>
{/* <CardActions>
<Button size="small">Share</Button>
<Button size="small">Learn More</Button>
</CardActions> */}
</Card>
);
}
export default Jobs;
i tried to do use the and flexbox instead of printing out the items everything they delete all the items in the webpage instead.
Try wrapping your parent component in Grid
import * as React from 'react';
import Jobs from './Jobs'
import { Grid } from '#mui/material';
const jobs1=[{
company:"razer",
jobtitle:"Data engineer",
},{
company:"Huawei",
jobtitle:"Data Engineer"
},{
company:"Huawei",
jobtitle:"Data Engineer"
}]
export default function Header() {
return (
<Grid container mt={2} spacing={2}>
jobs1.map((jobs1)=>(
<Jobs jobs1={jobs1}/>
))
</Grid>
)
}
Next: Wrapping you child in Grid
import React from 'react';
import Card from '#mui/material/Card';
import CardContent from '#mui/material/CardContent';
import CardMedia from '#mui/material/CardMedia';
import img from './image/Razer.png'
import Typography from '#mui/material/Typography';
import {Grid,Box } from '#mui/material'
const Jobs = ({jobs1: {company, jobtitle}})=>{
return (
<Grid item xs={4}>
<Card sx={{ maxWidth: 300 }}>
<CardMedia
component="img"
height="100"
sx={{ width: 100}}
image={img}
alt="razer"
/>
<CardContent>
<Typography style={{color: '#bdbdbd'}}gutterBottom variant="h9" component="div">
{company}
</Typography>
<Typography component="div">
<Box sx={{ fontWeight: 'bold' }}>
{jobtitle}
</Box>
</Typography>
<Grid container direction="row" alignItems="center" gap={14}>
<Typography sx={{fontStyle: 'italic'}} style={{color: '#bdbdbd'}}inline variant="body1" align="left">
1 minute ago
</Typography>
<Typography inline variant="body1" >
<Box sx={{ borderRadius: '12px', p: 1,opacity:"0.5", border: '0.5px solid',color: "#64dd17",bgcolor:"#ccff90" }}>Data</Box>
</Typography>
</Grid>
<Typography style={{color: '#bdbdbd'}}>____________________________</Typography>
</CardContent>
{/* <CardActions>
<Button size="small">Share</Button>
<Button size="small">Learn More</Button>
</CardActions> */}
</Card>
</Grid>
);
}
export default Jobs;
For more information you can check the Grid System of MUI

How can I fix this problem while mapping in a card?

import * as React from 'react';
import Card from '#mui/material/Card';
import CardActions from '#mui/material/CardActions';
import CardContent from '#mui/material/CardContent';
import CardMedia from '#mui/material/CardMedia';
import Button from '#mui/material/Button';
import Typography from '#mui/material/Typography';
export default function MediaCard() {
return (
<Card sx={{ maxWidth: 345 }}>
{data.map((item) =>(
<CardMedia component="img" height="140" image={`${data.img}`} alt="home image looding......"/>
<CardContent>
<Typography gutterBottom variant="h5" component="div">{data.name}</Typography>
<Typography variant="body2" color="text.secondary">{data.description}</Typography>
</CardContent>
))}
<CardActions>
<Button size="small">Share</Button>
<Button size="small">Learn More</Button>
</CardActions>
</Card>
);
}
let data =[
{
name:'Property for sale',
img: '/home1.jpg',
description: 'This House is 5.5 cent with square feet area of 2000 and 3 km from palakkad city ;100 meters from main road.',
},
{
name:'Property for Rent',
img: '/home2.jpg',
title: 'Burger',
},
{
name:'Property for sale',
img: '/home3.jpg',
title: 'Camera',
},
{
name:'Property for Rent',
img: '/home4.jpg',
title: 'Coffee',
},
{
name:'Property for sale',
img: '/home5.jpg',
title: 'Hats',
},
{
name:'Property for Rent',
img: '/home7.jpg',
title: 'Honey',
},
];
I can't get the details of mapped items in the card. While doing this I had found an error
"ERROR in src\Components\Card\Card.js
Line 17:6: Parsing error: Adjacent JSX elements must be wrapped in an enclosing tag. Did you want a JSX fragment <>...</>? (17:6)."
I had used material UI and I don't understand whether the mistake had happened.
It should be wrapped in a parent element. e.g
export default function MediaCard() {
return (
<div>
<Card sx={{ maxWidth: 345 }}>
{data.map((item) =>(
<CardMedia component="img" height="140" image={`${data.img}`} alt="home image looding......"/>
<CardContent>
<Typography gutterBottom variant="h5" component="div">{data.name}</Typography>
<Typography variant="body2" color="text.secondary">{data.description}</Typography>
</CardContent>
))}
<CardActions>
<Button size="small">Share</Button>
<Button size="small">Learn More</Button>
</CardActions>
</Card>
</div>
);
}
Change this
{data.map((item) =>(
<CardMedia component="img" height="140" image={`${data.img}`} alt="home image looding......"/>
<CardContent>
<Typography gutterBottom variant="h5" component="div">{data.name}</Typography>
<Typography variant="body2" color="text.secondary">{data.description}</Typography>
</CardContent>
))}
to this
{data.map((item) =>(
<>
<CardMedia component="img" height="140" image={`${data.img}`} alt="home image looding......"/>
<CardContent>
<Typography gutterBottom variant="h5" component="div">{data.name}</Typography>
<Typography variant="body2" color="text.secondary">{data.description}</Typography>
</CardContent>
</>
))}
You should have wrapped the inner element of the map in an empty Tag and also wrap the whole component in an <> </> this empty tag

Image is not loading in CardMedia in React

I have used an image as an object with a http link and also and image from the project source folder but it is not showing in the project.
Card and everything is loading up but image is not loading.
image location is public/assests/image
App.js
import React from 'react';
import Products from './components/Products/Products';
function App(props) {
return (
<div>
<Products />
</div>
);
}
export default App;
Products.jsx
import React from 'react';
import { Grid } from '#material-ui/core';
import Product from './Product/Product';
const products = [
{ id: 1, name: 'Shoes', description: 'Running shoes.', price: '$100', image: 'assests/image/run-asics-running-shoes-1636736175.jpeg' },
{ id: 2, name: 'Macbook', description: 'Apple Macbook', price: '$120', image: 'https://images.app.goo.gl/b8T9vFGUzQeHvRmM9' },
]
const Products = () => {
return (
<main>
<Grid container justify="center" spacing={4}>
{products.map((product) => (
<Grid item key={product.id} xs={12} sm={6} md={4} lg={3}>
<Product product={product} />
</Grid>
))}
</Grid>
</main>
)
}
export default Products;
Product component holding all the products
Product.jsx
import React from 'react';
import { Card, CardMedia, CardContent, CardActions, Typography, IconButton } from '#material-ui/core';
import { AddShoppingCart } from '#material-ui/icons';
import useStyles from './styles';
const Product = ({ product }) => {
const classes = useStyles();
return (
<div>
<Card className={classes.root}>
<CardMedia className={classes.media} image='product.image' title={product.name} />
<CardContent>
<div className={classes.cardContent}>
<Typography variant="h5" gutterBottom>
{product.name}
</Typography>
<Typography variant="h5" >
{product.price}
</Typography>
</div>
<Typography variant="body2" color="textSecondary">
{product.description}
</Typography>
</CardContent>
<CardActions disableSpacing className={classes.cardActions}>
<IconButton aria-label="Add to Cart">
<AddShoppingCart />
</IconButton>
</CardActions>
</Card>
</div>
);
}
export default Product;
You're setting image='product.image', which sets it to the literal string "product.image". Perhaps you meant image={product.image}?
<CardMedia
className={classes.media}
image={product.image}
title={product.name}
/>
In products array image with id:1 is ok
but image with id:2 does not contain correct image with HTTP link it should change to correct image link (in google image, right-click:copy image address)
Inside Product.js file, change image property of CardMedia to image={product.image} and it's style={{minHeight:300}}
Products:
const products = [
{ id: 1, name: 'Shoes', description: 'Running shoes.', price: '$100', image: 'assets/image/run.jpeg' },
{ id: 2, name: 'Macbook', description: 'Apple Macbook', price: '$120', image: '' },
]
Product
<CardMedia image={product.image} alt={product.name} style={{minHeight:300}}/>
Just replace the image='product.image' with image={product.image} and your problem solved
<CardMedia className={classes.media} image={product.image} title={product.name} />
<CardContent>
<div className={classes.cardContent}>
<Typography variant="h5" gutterBottom>
{product.name}
</Typography>
<Typography variant="h5" >
{product.price}
</Typography>
</div>
<Typography variant="body2" color="textSecondary">
{product.description}
</Typography>
</CardContent>

Button on card prevent new tab link from launching

I have a card component that has a green button on the top right, an image and text below it, I want to be able to click on anywhere on the card and for it to redirect to the url in the code; however if the green button is clicked, it should do nothing and not open the url. In the code below it doesn't do that, instead it always opens the url. I used href as it allows the left click on mouse to open in the same page, but the middle click to open a new tab while remaining in the same page which is very important.
Here is a link to a codeSandBox having the code: https://codesandbox.io/s/material-demo-forked-chy0d?file=/demo.js
Code:
export default function MediaCard() {
const classes = useStyles();
return (
<Card className={classes.root}>
<a href="sdfs">
<CardActionArea style={{ paddingTop: 50 }}>
<div
style={{
position: "absolute",
color: "white",
top: 8,
right: 8
}}
>
<Button
style={{ background: "green", color: "white" }}
onMouseDown={(e) => e.stopPropagation()}
size="small"
color="primary"
>
BUTTON
</Button>
</div>
<CardMedia
className={classes.media}
image="https://media.tarkett-image.com/large/TH_25121916_25131916_25126916_25136916_001.jpg"
title="Contemplative Reptile"
/>
<CardContent>
<Typography gutterBottom variant="h5" component="h2">
jnikn
</Typography>
<Typography variant="body2" color="textSecondary" component="p">
ijneodsk ipknv onfp o nf onopndsfc on pnojnc olpnoinoi hbib iubn
iujno nouno oijipj onoin ioio
</Typography>
</CardContent>
</CardActionArea>
</a>
</Card>
);
try this
import React from "react";
import { makeStyles } from "#material-ui/core/styles";
import Card from "#material-ui/core/Card";
import CardActionArea from "#material-ui/core/CardActionArea";
import CardActions from "#material-ui/core/CardActions";
import CardContent from "#material-ui/core/CardContent";
import CardMedia from "#material-ui/core/CardMedia";
import Button from "#material-ui/core/Button";
import Typography from "#material-ui/core/Typography";
const useStyles = makeStyles({
root: {
maxWidth: 345,
position: "relative"
},
media: {
height: 140
// marginTop: 50
}
});
export default function MediaCard() {
const classes = useStyles();
return (
<Card className={classes.root}>
<Button
style={{ zIndex:'1000',
position: "absolute",
top: 8,
right: 8,background: "green", color: "white",width:'70px',height:'30px' }}
onMouseDown={(e) => e.stopPropagation()}
size="small"
color="primary"
>
BUTTON
</Button>
<a href="sdfs">
<CardActionArea style={{ paddingTop: 50 }}>
<CardMedia
className={classes.media}
image="https://media.tarkett-image.com/large/TH_25121916_25131916_25126916_25136916_001.jpg"
title="Contemplative Reptile"
/>
<CardContent>
<Typography gutterBottom variant="h5" component="h2">
jnikn
</Typography>
<Typography variant="body2" color="textSecondary" component="p">
ijneodsk ipknv onfp o nf onopndsfc on pnojnc olpnoinoi hbib iubn
iujno nouno oijipj onoin ioio
</Typography>
</CardContent>
</CardActionArea>
</a>
</Card>
);
}

Material-UI Responsive Cards

I'm in the process of testing out Material-UI. I've been using Bootstrap for a long time, but am interested in adapting some React projects to Material-UI. Something I've been trying to figure out is how to create responsive cards in Material-UI. I've leaned pretty heavily on Bootstraps responsive containers in the past, so I can't understand why my cards expand with the page but don't shrink as the window is condensed...If we're meant to write custom css for this I'm cool with that, just need to be pointed in the right direction.
Questions:
Are responsive cards in Material-UI out of the gate a thing?
Or are we meant to write the css to make the cards responsive?
If so, where can I learn to do that? (leaned on Bootstrap a lot in the past)
Thanks in advance for your help!
...
useStyles = () => makeStyles(theme => ({
root: {
flexGrow: 1,
},
paper: {
padding: theme.spacing.unit,
textAlign: "center",
color: theme.palette.text.secondary,
marginBottom: theme.spacing.unit
},
}));
render() {
const {baseData} = this.state;
const {hfcMetrics} = this.state;
const {stateMember} = this.state;
const {stateName} = this.state;
const {states} = this.state;
const {lineVizData} = this.state;
const classes = this.useStyles();
return (this.state.doneLoading === false ? (
<div className={classes.root}>
<Grid container>
<Grid item xs={12}>
<ReactLoading type={"spinningBubbles"} color={"black"} height={'10%'}
width={'10%'} id='spinner'/>
</Grid>
</Grid>
</div>
)
:
(stateMember === true ?
<div className={classes.root}>
<Grid container spacing={3}>
<Grid item xs={12}>
<Card>
<CardHeader
title="Options"
/>
<CardContent style={{ width: '100%', height: 300 }}>
<LineViz
data={lineVizData}
state={stateName}
source='compareTool'
states={states}
vizKey={this.state.vizKey}
/>
</CardContent>
<CardActions>
<Button size="small" color="primary">
Share
</Button>
<Button size="small" color="primary">
Learn More
</Button>
</CardActions>
</Card>
</Grid>
<Grid item xs={6}>
<Card ref={this.elRef}>
<CardHeader
title="Comparison Analysis"
action={
<ButtonGroup variant="text" color="primary" aria-label="text primary button group">
<YearDropDown2
year={this.state.year}
handleChange={this.toggleYear}
/>
<IconButton color='default' component="span"
onClick={() => this.resetStateToggle()}><AutorenewRoundedIcon/></IconButton>
</ButtonGroup>
}
/>
<CardContent style={{padding: 0}}>
<DataCompareTable
data={this.state.compareData}
metric={this.state.metric}
stateName={this.state.stateName}
compareCount={this.state.compareCount}
handleChange={this.toggleStateName}
toggleOne={this.state.toggleOne}
toggleTwo={this.state.toggleTwo}
toggleThree={this.state.toggleThree}
/>
</CardContent>
</Card>
</Grid>
<Grid item xs={6}>
<Paper className={classes.paper}>xs=6</Paper>
</Grid>
</Grid>
</div>
: ''
)
)
}
}
export default CompareTool
Thank you for your question
To answer your questions:
1) not that I know of
2) no you dont have to write alot of css, but yes some css in the usestyles
3) Below I explained in detail the css you have to write, in your code. You are using inline styles and useStyles at the same time, try putting all your styles in the usestyle hook API instead and make it responsive. Hope this helps let me know if I need to make it clearer. Thank you
as far as I know, you can use the "useMediaQuery" hook to make your design responsive.
Here is the component from the material UI Card component page, I only added the useTheme and useMediaQuery imports, and added a medium breakpoint inside useStyle under classes.root Here is a useful link
on "useMediaQuery"
https://material-ui.com/components/use-media-query/#usemediaquery
import { useTheme } from "#material-ui/styles";
import useMediaQuery from "#material-ui/core/useMediaQuery";
const useStyles = makeStyles(theme => ({
root: {
maxWidth: 345,
[theme.breakpoints.down("md")] : {
maxWidth: 200
}
},
media: {
height: 140
}
}));
const Card = () => {
const classes = useStyles();
const theme = useTheme();
const matches = useMediaQuery(theme.breakpoints.up("sm"));
return (
<Card className={classes.root}>
<CardActionArea>
<CardMedia
className={classes.media}
title="Contemplative Reptile"
/>
<CardContent>
<Typography gutterBottom variant="h5" component="h2">
Lizard
</Typography>
<Typography variant="body2" color="textSecondary" component="p">
Lizards are a widespread group of squamate reptiles, with over 6,000
species, ranging across all continents except Antarctica
</Typography>
</CardContent>
</CardActionArea>
<CardActions>
<Button size="small" color="primary">
Share
</Button>
<Button size="small" color="primary">
Learn More
</Button>
</CardActions>
</Card>
);
}
Hope this helps
You can wrap your card in a container with responsive options or use grids, which is very convenient.

Resources