基于react做的消息轮播

39 阅读2分钟

又来组件积累了,为了少加班,今天写了个消息单条轮播,先记下笔记,以后方便复制。经常用于首页消息通知。

import { useEffect, useState, useRef,useLayoutEffect } from 'react';
import styles from './comp.module.less';
const NoticeCarousel = props => {
  const { notifications = [] } = props;
  console.log('notifications',notifications)
  let timer;
  let activeIndex = 0;
  let shouldReset = false;
  let moving = false;
  const interval = 4000;
  const animationDuration = 500;
  const swiperHeight = 30;
  const noticeBarList = useRef();
  const [firstItem, setFirstItem] = useState(null);
  const listRef = useRef();
  const [list, setList] = useState([]);

  function transitionEnd() {
    moving = false;
    const currentIndex = activeIndex;
    if (shouldReset) {
      shouldReset = false;
      swiperDoTransition(currentIndex, 0);
      if(!firstItem) return;
      firstItem.style.transform = '';
    }
  }

  function next() {
    const nextIndex = activeIndex + 1;
    const isLastItem = nextIndex >= list.length;
    moving = true;
    swiperDoTransition(nextIndex, animationDuration);
    if (isLastItem) {
      activeIndex = 0;
      shouldReset = true;
      if(!firstItem) return;
      firstItem.style.transform = `translate3d(0, calc(${nextIndex} * ${swiperHeight}px), 0)`;
    } else {
      activeIndex += 1;
    }
  }

  function autoPlay() {
    console.log('list', list);
    console.log('list', list.length);

    if (list.length <= 1) return;
    timer && clearInterval(timer);
    timer = setInterval(() => {
      !moving && next();
    }, interval + animationDuration);
  }

  function swiperDoTransition(index, duration) {
    const transformY = `translate3d(0, calc(-${index} * ${swiperHeight}px), 0)`;
    if(!noticeBarList.current) return;
    noticeBarList.current.style.transitionDuration = `${duration}ms`;
    noticeBarList.current.style.transform = transformY;
  }
  useLayoutEffect(() => {
    setFirstItem(noticeBarList.current.firstElementChild);
    setList(noticeBarList.current.children);
  }, []);
  useLayoutEffect(() => {
    timer = null;
    noticeBarList.current.addEventListener('transitionend', transitionEnd);
    setTimeout(() => {
      autoPlay();
    }, 2000);
    return () => {
      timer && clearInterval(timer);
    };
  }, [list]);

  return (
    <div className={styles['page-container']}>
      <div className={styles['noticebar']}>
        <div className={styles['noticebar-list-container']}>
          <div className={styles['noticebar-list']} ref={noticeBarList}>
            {notifications?.map(i => (
              <div className={styles['noticebar-item']} key={i.messageId} ref={listRef}>
                <div className={styles['custom-notice-item']}>
                  <p className={styles.notice} dangerouslySetInnerHTML={{ __html: i.content }} />
                </div>
              </div>
            ))}
          </div>
        </div>
        {/* <div className={styles['noticebar-close-icon']}>
          <img
            src=""
            alt=""
          />
        </div> */}
      </div>
    </div>
  );
};
export default NoticeCarousel;



样式

/*** 通知栏轮播部分 ***/
.page-container {
  width: 100%;
  overflow: hidden;
  .noticebar {
    width: 100%;
    --height: 30px;
    box-sizing: border-box;
    color: #2e3346;
    font-size: 14px;
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
    .noticebar-list-container {
      width: 100%;
      height: var(--height);
      overflow: hidden;
      flex-shrink: 0;
    }
  }
  .noticebar-close-icon {
    width: 15px;
    height: 15px;
  }
  img {
    display: block;
    width: 100%;
    height: 100%;
  }
  .noticebar-item {
    height: var(--height);
    align-items: center;
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
    .notice {
      text-overflow: ellipsis;
      white-space: nowrap;
      overflow: hidden;
    }
  }
  .custom-notice-item {
    width: 100%;
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
    flex-shrink: 0;
    line-height: 30px;
  }
}