Commit 3310254d authored by 周宏民's avatar 周宏民

feat: 熊猫智慧水务一体化解决方案

parent 4ab564dd
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
This diff was suppressed by a .gitattributes entry.
...@@ -62,8 +62,10 @@ const BottomItem = props => { ...@@ -62,8 +62,10 @@ const BottomItem = props => {
return reArr; return reArr;
}; };
useEffect(() => { useEffect(() => {
setTimeout(() => {
const arr = Array(44).fill({}); const arr = Array(44).fill({});
setListData(spliceArr(arr, 2)); setListData(spliceArr(arr, 2));
}, 1000);
}, []); }, []);
return ( return (
<div className={classNames(styles.bottom_item, 'demo_bottom_item')}> <div className={classNames(styles.bottom_item, 'demo_bottom_item')}>
...@@ -103,7 +105,7 @@ const BottomItem = props => { ...@@ -103,7 +105,7 @@ const BottomItem = props => {
<div className={styles.center}> <div className={styles.center}>
<DataCarousel <DataCarousel
gap={1} gap={1}
autoplay={0} autoplay={3000}
itemHeight={270} itemHeight={270}
list={listData} list={listData}
renderItem={renderCol} renderItem={renderCol}
......
...@@ -3,11 +3,11 @@ ...@@ -3,11 +3,11 @@
* @Author: hongmye * @Author: hongmye
* @Date: 2023-12-26 18:34:42 * @Date: 2023-12-26 18:34:42
*/ */
import { FullscreenExitOutlined, FullscreenOutlined } from '@ant-design/icons';
import { Button } from 'antd'; import { Button } from 'antd';
import React, { useMemo, useEffect, useState, memo } from 'react'; import React, { useMemo, useEffect, useState, memo } from 'react';
import { schemeData } from '../configData'; import { schemeData } from '../configData';
import styles from './index.less'; import styles from './index.less';
let timer;
const LeftItem = props => { const LeftItem = props => {
const { title = '', setSelectKey, selectKey } = props; const { title = '', setSelectKey, selectKey } = props;
...@@ -16,6 +16,18 @@ const LeftItem = props => { ...@@ -16,6 +16,18 @@ const LeftItem = props => {
setSelectKey(row.title); setSelectKey(row.title);
} }
}; };
const onMouseEnter = (item, index) => {
if (timer) {
timer && clearTimeout(timer);
}
timer = setTimeout(() => {
props.onChangeScheme && props.onChangeScheme(item, index);
}, 500);
};
const onMouseLeave = (item, index) => {
timer && clearTimeout(timer);
};
useEffect(() => {}, []); useEffect(() => {}, []);
return ( return (
<div className={styles.left_item}> <div className={styles.left_item}>
...@@ -29,8 +41,9 @@ const LeftItem = props => { ...@@ -29,8 +41,9 @@ const LeftItem = props => {
className={styles.l_list_item} className={styles.l_list_item}
key={item.title} key={item.title}
type={item.title} type={item.title}
selectType={selectKey === item.title ? 'select' : ''}
onClick={() => onSelect(item)} onClick={() => onSelect(item)}
onMouseEnter={() => onMouseEnter(item, index)}
onMouseLeave={() => onMouseLeave(item, index)}
> >
<div className={styles.l_list_item_title}>- {item.title} -</div> <div className={styles.l_list_item_title}>- {item.title} -</div>
<div className={styles.l_list_item_tip} /> <div className={styles.l_list_item_tip} />
......
...@@ -39,16 +39,16 @@ ...@@ -39,16 +39,16 @@
flex-wrap: wrap; flex-wrap: wrap;
} }
// .l_list_item:hover { .l_list_item:hover {
// opacity: 1; opacity: 1;
// top: -8px; top: -8px;
// } }
// .l_list_item:hover .l_list_item_tip { .l_list_item:hover .l_list_item_tip {
// opacity: 1; opacity: 1;
// } }
.l_list_item[selectType='select'] { .l_list_item[selectType='select'] {
opacity: 1; opacity: 1;
......
...@@ -3,53 +3,109 @@ ...@@ -3,53 +3,109 @@
* @Author: hongmye * @Author: hongmye
* @Date: 2023-12-26 18:34:42 * @Date: 2023-12-26 18:34:42
*/ */
import { FullscreenExitOutlined, FullscreenOutlined } from '@ant-design/icons';
import { Button } from 'antd'; import { Button } from 'antd';
import React, { useMemo, useEffect, useRef } from 'react'; import React, { useMemo, useEffect, useRef, useState, forwardRef, useImperativeHandle } from 'react';
import { cloneDeep, debounce } from 'lodash';
import classNames from 'classnames';
import SwiperCore, { Autoplay, Mousewheel, Navigation, Pagination, EffectCoverflow } from 'swiper/core';
import { Swiper, SwiperSlide } from 'swiper/react';
import leftIcon from '@/assets/demonstration/arrow_left.png';
import rightIcon from '@/assets/demonstration/arrow_right.png';
import { schemeData } from '../configData';
import 'swiper/components/navigation/navigation.min.css';
import 'swiper/components/pagination/pagination.min.css';
import 'swiper/components/effect-coverflow/effect-coverflow.min.css';
import 'swiper/swiper.min.css';
import styles from './index.less'; import styles from './index.less';
const boxWidth = 982; const autoplay = 0;
const boxHeight = 455; SwiperCore.use([Autoplay, Pagination, Navigation, Mousewheel, EffectCoverflow]);
const VideoItem = (props, ref) => {
const VideoItem = props => { const { selectKey = '' } = props;
const { selectKey = '', info = {} } = props; const swiperRef = useRef(null);
const contentRef = useRef(null); const onSlideToLoop = index => {
const onResize = () => { if (swiperRef.current) {
if (contentRef?.current) { swiperRef.current.slideToLoop(index);
const { clientWidth, clientHeight } = contentRef.current;
if (!boxWidth || !boxHeight) return;
const xScale = clientWidth / boxWidth;
const yScale = clientHeight / boxHeight;
const scale = Math.min(xScale, yScale);
const n = scale.toFixed(4);
} }
}; };
useImperativeHandle(ref, () => ({
onSlideToLoop,
}));
useEffect(() => { const listData = schemeData.filter(i => !!i.video);
window.addEventListener('resize', debounce(onResize, 300));
return () => { const renderItem = (row, index) => (
window.removeEventListener('resize', onResize); <div className={styles.video_box} key={row.title}>
}; <div className={styles.video_tip} type={row.title}>
}, []);
return (
<div className={styles.video_item}>
<div className={styles.video_box} ref={contentRef}>
<div className={styles.video_tip} type={selectKey}>
<span /> <span />
{selectKey} {row.title}
</div> </div>
<div className={styles.video_wrap}> <div className={styles.video_wrap}>
<video <video autoPlay="autoPlay" muted controls src={row.video} loop />
autoPlay="autoPlay" </div>
muted </div>
controls );
src="http://vfx.mtime.cn/Video/2021/07/10/mp4/210710122716702150.mp4" const onSwiper = type => {
/> if (swiperRef.current) {
if (type === 'left') {
swiperRef.current.slidePrev();
}
if (type === 'right') {
swiperRef.current.slideNext();
}
}
};
const onSlideChange = e => {
if (e.$el[0]) {
const videoDom = e.$el[0].querySelector('video');
videoDom.play && videoDom.play();
}
};
useEffect(() => {}, []);
return (
<div className={classNames(styles.video_list, 'dt_video_list')}>
<Swiper
slidesPerView="auto"
autoplay={{
delay: autoplay,
disableOnInteraction: false,
}}
loop
height={450}
direction="horizontal"
onSlideChange={onSlideChange}
effect="coverflow"
coverflowEffect={{
rotate: 50,
stretch: 10,
depth: 60,
modifier: 2,
slideShadows: false,
}}
mousewheel
onSwiper={swiper => {
swiperRef.current = swiper;
// 鼠标悬浮暂停效果
swiper.$el[0].addEventListener('mouseover', () => swiper.autoplay.stop());
// 鼠标移开后继续自动滚屏效果
swiper.$el[0].addEventListener('mouseleave', () => {
listData.length > 1 && autoplay * 1 ? swiper.autoplay.start() : '';
});
}}
>
{listData.map((item, index) => (
<SwiperSlide key={index} virtualIndex={index} onClick={() => {}}>
{renderItem(item, index + 1)}
</SwiperSlide>
))}
</Swiper>
<div className={classNames(styles.swiper_btn, styles.swiper_left)} onClick={() => onSwiper('left')}>
<img src={leftIcon} alt="" />
</div> </div>
<div className={classNames(styles.swiper_btn, styles.swiper_right)} onClick={() => onSwiper('right')}>
<img src={rightIcon} alt="" />
</div> </div>
</div> </div>
); );
}; };
export default VideoItem; export default forwardRef(VideoItem);
@imgSrc: '@/assets/demonstration'; @imgSrc: '@/assets/demonstration';
.video_item { .video_list {
width: 100%; width: 100%;
padding: 0 30px; padding: 0 30px;
height: calc(100% - 25px); height: calc(100% - 25px);
margin-top: 25px; margin-top: 25px;
background: url('@{imgSrc}/视频bg2.png') no-repeat center center; background: url('@{imgSrc}/视频bg2.png') no-repeat center center;
background-size: calc(100% - 30px) auto; background-size: calc(100% - 30px) auto;
position: relative;
.swiper_btn {
position: absolute;
width: 36px;
height: 40px;
cursor: pointer;
z-index: 100;
img {
width: 100%;
}
}
.swiper_btn:active {
opacity: 0.8;
}
.swiper_left {
left: 20px;
top: 50%;
transform: translateY(-50%);
}
.swiper_right {
right: 20px;
top: 50%;
transform: translateY(-50%);
}
.video_box { .video_box {
width: 100%; width: 100%;
...@@ -33,8 +64,10 @@ ...@@ -33,8 +64,10 @@
padding: 10px; padding: 10px;
video { video {
object-fit: fill;
width: 100%; width: 100%;
height: 100%; height: 100%;
border-radius: 10px;
} }
} }
...@@ -78,3 +111,17 @@ ...@@ -78,3 +111,17 @@
} }
} }
:global {
.dt_video_list {
.swiper-slide {
width: 100%;
}
.swiper-container {
width: 1100px;
height: 100%;
}
}
}
\ No newline at end of file
...@@ -4,26 +4,34 @@ ...@@ -4,26 +4,34 @@
* @Author: hongmye * @Author: hongmye
* @Date: 2024-01-30 15:29:09 * @Date: 2024-01-30 15:29:09
*/ */
import gongshuiVideo from '@/assets/videos/供水.mp4';
import shuiliVideo from '@/assets/videos/水利.mp4';
import paishuiVideo from '@/assets/videos/排水.mp4';
import jieshuiVideo from '@/assets/videos/节水.mp4';
const schemeData = [ const schemeData = [
{ {
title: '供水产品', title: '供水产品',
background: '', background: '',
url: '', url: '',
video: gongshuiVideo,
}, },
{ {
title: '水利产品', title: '水利产品',
background: '', background: '',
url: '', url: '',
video: shuiliVideo,
}, },
{ {
title: '排水产品', title: '排水产品',
background: '', background: '',
url: '', url: '',
video: paishuiVideo,
}, },
{ {
title: '节水产品', title: '节水产品',
background: '', background: '',
url: '', url: '',
video: jieshuiVideo,
}, },
]; ];
const productData = [ const productData = [
......
...@@ -21,9 +21,10 @@ import { platformData } from './components/configData'; ...@@ -21,9 +21,10 @@ import { platformData } from './components/configData';
const boxWidth = 1920; const boxWidth = 1920;
const boxHeight = 911; const boxHeight = 911;
const Demonstration = props => { const Demonstration = props => {
const onLineUrl = window.globalConfig?.mainserver || 'https://panda-water.cn/';
const showFullScreen = true; const showFullScreen = true;
const videoRef = useRef(null);
const [selectKey, setSelectKey] = useState('供水产品'); const [selectKey, setSelectKey] = useState('供水产品');
const [scale, setScaley] = useState(1);
const [boxSize, setBoxSize] = useState({ const [boxSize, setBoxSize] = useState({
scale: 1, scale: 1,
boxHeight: 911, boxHeight: 911,
...@@ -38,6 +39,13 @@ const Demonstration = props => { ...@@ -38,6 +39,13 @@ const Demonstration = props => {
handleFullScreen && handleFullScreen(); handleFullScreen && handleFullScreen();
} }
}; };
// 切换方案视频
const onChangeScheme = (item, index) => {
if (videoRef.current) {
videoRef.current.onSlideToLoop(index);
}
};
const renderCenter = useMemo( const renderCenter = useMemo(
() => () =>
platformData.map(col => { platformData.map(col => {
...@@ -123,10 +131,10 @@ const Demonstration = props => { ...@@ -123,10 +131,10 @@ const Demonstration = props => {
<div className={styles.row}> <div className={styles.row}>
<div className={styles.row_l}> <div className={styles.row_l}>
<LeftItem setSelectKey={setSelectKey} selectKey={selectKey} /> <LeftItem setSelectKey={setSelectKey} selectKey={selectKey} onChangeScheme={onChangeScheme} />
</div> </div>
<div className={styles.row_c}> <div className={styles.row_c}>
<VideoItem selectKey={selectKey} setSelectKey={setSelectKey} /> <VideoItem ref={videoRef} selectKey={selectKey} setSelectKey={setSelectKey} onLineUrl={onLineUrl} />
</div> </div>
<div className={styles.row_r}> <div className={styles.row_r}>
<RightItem /> <RightItem />
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment