Я работаю в простом проекте, но у меня возникли некоторые проблемы с слайдером изображений, который я сделал для него.
Проблема
контейнер потерял свое соотношение после 2500px w
изображение не сохраняет его соотношение сторон
У меня недостаточно репутации, чтобы правильно размещать изображения
Изображение не масштабируется вместе с контейнером.
Когда контейнер становится больше, ползунок переходит в сторону, а не в крышку
когда контейнер становится меньше, ползунок не сохраняет его пропорции.
код объяснения
Я использую React.js, в данном случае мой собственный функциональный компонент Карусель.
Использование CSS-модулей.
с использованием CSS Grid.
Код карусели:
CSS
* {
padding: 0;
margin: 0;
box-sizing: border-box;
}
.CarouselContainer {
/* width: 600px;
overflow: hidden;
max-width: 100%; */
}
.CarouselSlide {
display: flex;
}
img {
object-fit: cover;
/* width: 100%; */
height: 100%;
}
.PrevBtn {
position: absolute;
top: 50%;
z-index: 10;
left: 5%;
font-size: 30px;
color: white;
cursor: pointer;
background-color: transparent;
border: none;
}
.PrevBtn:disabled {
opacity: 0.3;
}
.PrevBtn:focus {
outline: 0;
}
.NextBtn {
position: absolute;
top: 50%;
z-index: 10;
right: 5%;
font-size: 30px;
color: white;
cursor: pointer;
background-color: transparent;
border: none;
}
.NextBtn:disabled {
opacity: 0.3;
}
.NextBtn:focus {
outline: 0;
}
.Like {
position: absolute;
top: 2%;
z-index: 10;
right: 2%;
color: white;
cursor: pointer;
font-size: 50px;
background-color: transparent;
border: none;
color: rgba(255, 90, 95, 0.9);
}
.Like:focus {
outline: 0;
}
.Dislike {
position: absolute;
top: 2%;
z-index: 10;
right: 2%;
color: white;
cursor: pointer;
font-size: 50px;
background-color: transparent;
border: none;
color: rgba(8, 1, 1, 0.6);
}
.Dislike:focus {
outline: 0;
}
JavaScript:
import React, { useEffect, useRef, useState } from 'react';
import { FaArrowLeft, FaArrowRight, FaHeart } from 'react-icons/fa';
import classes from './carousel.module.css';
const Carousel = ({ id, likeState, onLike }) => {
// states
const [counter, setCounter] = useState(0);
const [nodeChildren, setNodeChildren] = useState();
// references
const carouselSlide = useRef();
const prevBtn = useRef();
const nextBtn = useRef();
// button actions
const nextImage = () => {
setNodeChildren(carouselSlide.current.childNodes[0].clientWidth);
carouselSlide.current.style.transform = 'transform 0.4s ease-in-out';
setCounter(counter + 1);
carouselSlide.current.style.transform = `translateX(${-nodeChildren *
counter}px)`;
};
const prevImage = () => {
setNodeChildren(carouselSlide.current.childNodes[0].clientWidth);
carouselSlide.current.style.transform = 'transform 0.4s ease-in-out';
setCounter(counter - 1);
carouselSlide.current.style.transform = `translateX(${-nodeChildren *
counter}px)`;
};
const likeHandler = () => {
onLike({ type: 'LIKE', id });
};
useEffect(() => {
carouselSlide.current.style.transform = `translateX(${-nodeChildren *
counter}px)`;
}, [counter, nodeChildren]);
let likeClasses = [classes.Dislike];
if (likeState.boolean) {
likeClasses = [classes.Like];
}
return (
<div>
<div className={classes.CarouselContainer}>
<button
className={classes.PrevBtn}
ref={prevBtn}
onClick={prevImage}
disabled={counter === 0}
>
<FaArrowLeft />
</button>
<button
className={classes.NextBtn}
ref={nextBtn}
onClick={nextImage}
disabled={counter === 4}
>
<FaArrowRight />
</button>
<button className={likeClasses} onClick={likeHandler}>
<FaHeart />
</button>
<div className={classes.CarouselSlide} ref={carouselSlide}>
<img src="images/1.jpg" alt="" />
<img src="images/2.jpg" alt="" />
<img src="images/3.jpg" alt="" />
<img src="images/4.jpg" alt="" />
<img src="images/5.jpg" alt="" />
</div>
</div>
</div>
);
};
export default Carousel;
Код контейнера
CSS
* {
font-family: "Roboto", sans-serif;
box-sizing: border-box;
outline-style: none;
}
:focus {
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
}
span {
vertical-align: baseline;
position: relative;
top: -0.4em;
font-size: 1.3vw;
}
.Container {
width: 80%;
display: grid;
/* grid-template-columns: minmax(100px, 3fr) minmax(150px, 7fr); */
grid-template-columns: 30% 70%;
grid-template-rows: 24vw;
grid-template-areas: "image info";
margin: 25px auto;
}
.Images {
background-color: lightcyan;
grid-area: image;
overflow: hidden;
max-height: 600px;
max-width: 600px;
position: relative;
}
/* .Images:first-child {
max-height: 100%;
display: none;
} */
/* .Carousel {
max-width: 100%;
height: auto;
} */
.Info {
display: grid;
grid-template-columns: 1fr;
grid-template-rows: 7fr 3fr 2fr;
grid-template-areas:
"text"
"value"
"list";
}
.Info h1 {
color: #1598f6;
font-size: 1.8vw;
margin: 0.5em 1em 0.2em;
}
.Info h1 + p {
font-size: 1.3vw;
color: #747373;
margin: 1em 1.2em;
}
.Info h1 + p + p {
font-size: 1.4vw;
color: #605e5e;
margin: 1em 1.2em 0em;
line-height: 1.5em;
}
.Price {
display: grid;
grid-template-columns: 1fr 2fr;
}
.Price p:first-child,
.InputPrice {
/* margin: 0 auto; */
margin-left: 0.8em;
font-size: 2.3vw;
color: #000000;
font-weight: bold;
align-self: center;
}
.Price p:nth-child(2) {
font-size: 1.6vw;
color: #1598f6;
font-weight: bold;
align-self: center;
margin-left: 1vw;
}
.Description {
background-color: #e9e9e9;
display: grid;
grid-template-columns: 2fr 1fr;
grid-template-areas: "info contact";
}
.Description ul {
grid-area: info;
width: 80%;
display: flex;
list-style-type: none;
padding: 0;
justify-items: center;
margin: auto 1em;
}
.Description li {
font-size: 1.2vw;
color: #000000;
font-weight: bold;
justify-self: center;
margin: 0 auto;
}
.Description li > span {
font-size: 1vw;
}
.Description p {
grid-area: contact;
margin: auto;
font-size: 1.6vw;
color: #1598f6;
font-weight: bold;
cursor: pointer;
}
.InputPrice {
display: grid;
grid-template-columns: 4fr 1fr;
align-items: center;
border: none;
}
.InputPrice input,
button {
border: none;
height: 3vw;
font-size: 1vw;
border: 1px solid #757575;
border-right-color: transparent;
border-top-left-radius: 3px;
border-bottom-left-radius: 3px;
}
.InputPrice button {
color: black;
background-color: rgba(21, 152, 246, 0.8);
}
.InputPrice button:disabled {
background-color: rgba(238, 238, 238, 1);
}
@media only screen and (max-width: 800px) {
.Container {
width: 80%;
grid-template-columns: 1fr;
grid-template-rows: 80vw 80vw;
grid-template-areas:
"image"
"info";
margin: 0 auto;
position: relative;
}
.Info {
grid-template-columns: 100%;
grid-template-rows: 7fr 3fr;
grid-template-areas:
"text"
"value";
}
.Info h1 {
margin: 0.5em auto 0.2em;
width: 95%;
font-size: 4.5vw;
text-align: left;
}
.Info h1 + p {
font-size: 3vw;
width: 95%;
margin: 1em auto;
}
.Info h1 + p + p {
font-size: 3.5vw;
width: 95%;
margin: 1em auto;
text-align: justify;
}
.Price {
position: absolute;
top: 70vw;
text-shadow: -1px 0 black, 0 1px black, 1px 0 black, 0 -1px black;
}
.Price p:first-child {
font-size: 3.3vw;
color: white;
}
.Price p:nth-child(2) {
font-size: 2.5vw;
color: white;
margin-left: 1em;
}
.Description {
background-color: transparent;
}
.Description ul {
width: 100%;
position: absolute;
top: -1.5em;
left: -2em;
padding: 0;
}
.Description li {
font-size: 2.4vw;
color: white;
text-shadow: -1px 0 black, 0 1px black, 1px 0 black, 0 -1px black;
}
.Description li > span {
font-size: 2vw;
}
.Description p {
margin: 2em 0;
font-size: 3.6vw;
}
}
код JavaScript
import React, { useReducer, useState, useRef } from 'react';
import { FaMapMarkerAlt } from 'react-icons/fa';
import classes from './Department.module.css';
import Carousel from '../Carousel/carousel';
const Department = ({ price, mtrs, id, boolean }) => {
// reducers
const storeReducer = (state, action) => {
switch (action.type) {
case 'LIKE':
localStorage.setItem(action.id, !state.boolean);
return { ...state, boolean: !state.boolean };
case 'NEW_PRICE':
console.log(state);
localStorage.setItem(`${action.id}-price`, action.price);
return { ...state, price: action.price };
default:
return state;
}
};
const initialState = {
boolean,
price,
};
// states
const [inputValue, setInputValue] = useState(false);
const [store, dispatch] = useReducer(storeReducer, initialState);
const [inputIsValid, setInputIsValid] = useState(false);
// ref
const priceRef = useRef();
// handlers
const inputHandler = () => setInputValue(!inputValue);
const inputValidationHandler = event => {
if (
isNaN(parseInt(priceRef.current.value)) ||
event.target.value.trim() === ''
) {
setInputIsValid(false);
} else {
setInputIsValid(true);
}
};
const priceChangeHandler = () => {
console.log(parseInt(priceRef.current.value));
dispatch({
type: 'NEW_PRICE',
id,
price: parseInt(priceRef.current.value),
});
setInputValue(false);
};
// price render variable
let priceEleType = (
<p onClick={inputHandler}>
US
{store.price
.toLocaleString('en-US', {
style: 'currency',
currency: 'USD',
})
.slice(0, -3)}
</p>
);
if (inputValue)
priceEleType = (
<div className={classes.InputPrice}>
<input
type="text"
placeholder="Price"
maxLength={7}
ref={priceRef}
onChange={inputValidationHandler}
style={{
backgroundColor: inputIsValid
? 'transparent'
: 'rgba(255, 90, 95, 0.2)',
}}
autoFocus
/>
<button
type="button"
onClick={priceChangeHandler}
disabled={!inputIsValid}
>
Add
</button>
</div>
);
return (
<div className={classes.Container}>
<div className={classes.Images}>
<Carousel
id={id}
likeState={store}
onLike={dispatch}
className={classes.Carousel}
/>
{/* <img src="images/2.jpg" alt="" />
<img src="images/2.jpg" alt="" /> */}
</div>
<div className={classes.Info}>
<div className={classes.Text}>
<h1>
Si vas a utilizar un pasaje de Lorem Ipsum, necesitas estar seguro
</h1>
<p>
<FaMapMarkerAlt />
{` Juan Francisco Seguí 3900, Palermo Chico, Pal...`}
</p>
<p>
Lorem Ipsum es simplemente el texto de relleno de las imprentas y
archivos de texto Lorem Ipsum ha sido el texto de relleno estándar
de las industrias desde el año 1500, cuando un impresor (N. del T.
persona que se dedica a la imprenta)
</p>
</div>
<div className={classes.Price}>
{priceEleType}
<p>
US$ $/m<span>2</span>{' '}
{parseInt(store.price / mtrs).toLocaleString()}
</p>
</div>
<div className={classes.Description}>
<ul>
<li>
{mtrs}m<span>2</span>
</li>
<li>3 dormitorios</li>
<li>2 baños</li>
<li>2 cocheras</li>
</ul>
<p className={classes.Contact}>CONTACTAR</p>
</div>
</div>
</div>
);
};
export default Department;
Очень длинный код, но я думаю, что важная часть - это css.
Попытка
Я пробовал с max / min-height / width, определяя ширину-высоту, подгонку объекта, создавая новые контейнеры, перемещая ползунок в коде jsx.
Честно говоря, я попробовал почти каждый ответ в StackOverflow.
Ожидаемое поведение
контейнер работает, но изображение не сохраняет соотношение сторон
У меня недостаточно репутации, чтобы правильно размещать изображения
Изображение масштабируется и сохраняет его пропорции.
Когда ширина экрана превышает 2500 пикселей, контейнер не разбивается.
Любая помощь будет удивительной.
весь код мой, я не могу предоставить библиотеку слайдеров, так как я не использовал одну