SwiperGroup.tsx 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. "use client";
  2. import Box from "@/components/Box";
  3. import { FC, PropsWithChildren, ReactNode, useRef } from "react";
  4. import { Swiper, SwiperClass, SwiperSlide } from "swiper/react";
  5. import { Category, GameListRep } from "@/api/home";
  6. import { CardProps } from "@/components/Card/Card";
  7. import GroupCard, { GroupProps } from "@/components/Card/GroupCard";
  8. import clsx from "clsx";
  9. import { useTranslations } from "next-intl";
  10. import styles from "./style.module.scss";
  11. type TodoHandler = (item: Category) => void;
  12. interface SwiperGroupProps extends GroupProps, CardProps {
  13. group: Category;
  14. page?: number;
  15. // style
  16. bg?: boolean;
  17. visibleTitle?: boolean;
  18. // todos
  19. visibleTodos?: boolean;
  20. todoHandler?: TodoHandler;
  21. // icon
  22. offsetX?: number;
  23. offsetY?: number;
  24. // SwiperSlide
  25. slideRender?: (data: GameListRep[]) => ReactNode;
  26. }
  27. const HomeSwiper: FC<PropsWithChildren<SwiperGroupProps>> = (props) => {
  28. const {
  29. group,
  30. page = 3,
  31. todoHandler,
  32. bg = true,
  33. visibleTitle = true,
  34. visibleTodos = true,
  35. slideRender,
  36. ...other
  37. } = props;
  38. const t = useTranslations("HomePage");
  39. const swiperRef = useRef<SwiperClass | null>(null);
  40. const bodyClass = clsx(
  41. {
  42. ["bg-[#1a1a1a]"]: bg,
  43. },
  44. "mb-[0.11rem]"
  45. );
  46. if (!group.game_list) return;
  47. const lineNum = page * group.line_num;
  48. const gameList = group.game_list.slice(0, group.line_config_amount);
  49. const swiperData = Array(gameList.length ? Math.ceil(gameList.length / lineNum) : 0)
  50. .fill(0)
  51. .map((_, index) => {
  52. return {
  53. key: index,
  54. data: gameList.slice(index * lineNum, index * lineNum + lineNum).map(subItem => ({ ...subItem, category_name: group.category_name }))
  55. };
  56. });
  57. const prev = () => {
  58. swiperRef.current!.slidePrev();
  59. };
  60. const next = () => {
  61. swiperRef.current!.slideNext();
  62. };
  63. const iconClass = clsx(styles.mainLeftIcon, "pro-iconfont", `pro-${group.icon}`);
  64. return (
  65. <div>
  66. {visibleTitle && (
  67. <Box none>
  68. <div className={styles.mainTitle}>
  69. <div className={styles.mainLeft}>
  70. <div className={iconClass}></div>
  71. <div className={styles.marginLeftTitle}>{group.category_name}</div>
  72. <div className={styles.marginLeftTag}> {t("gameTag")}</div>
  73. </div>
  74. <img
  75. src=""
  76. alt=""
  77. />
  78. <div className={styles.mainRight}>
  79. {visibleTodos && (
  80. <div onClick={() => todoHandler && todoHandler(group)}>
  81. Todos
  82. <span className={styles.marginRightCount}>{group.to_tal}</span>
  83. </div>
  84. )}
  85. <div onClick={prev}>
  86. <span className="iconfont icon-xiangzuo1"></span>
  87. </div>
  88. <div onClick={next}>
  89. <span className="iconfont icon-xiangyou1"></span>
  90. </div>
  91. </div>
  92. </div>
  93. </Box>
  94. )}
  95. <Box none className={bodyClass}>
  96. <Box>
  97. <Swiper
  98. spaceBetween={10}
  99. slidesPerView={1}
  100. onSwiper={(swiper) => {
  101. swiperRef.current = swiper;
  102. }}
  103. loop
  104. >
  105. {swiperData?.map((data, index) => (
  106. <SwiperSlide key={data.key}>
  107. {slideRender ? (
  108. slideRender(data.data)
  109. ) : (
  110. <GroupCard
  111. data={data.data}
  112. {...other}
  113. groupType={group.bet_type}
  114. ></GroupCard>
  115. )}
  116. </SwiperSlide>
  117. ))}
  118. </Swiper>
  119. </Box>
  120. </Box>
  121. </div>
  122. );
  123. };
  124. export default HomeSwiper;