关于javascript:实现一个响应式的瀑布流组件

在图片网站中,通常是应用瀑布流的形式来显示图片,等宽度不等高,这样显得错落有致

明天 应用Vue3实现一个瀑布流组件,并公布为一个npm

  1. grid布局
  2. 反对响应式

参考自react-plock

1、window.matchMedia

window.matchMedia(mediaQueryString)办法返回一个MediaQueryList对象,示意指定css媒体查问字符串的后果

其中参数mediaQueryString示意CSS @media规定的任何媒体查问选项,比方:

  • min-widthmin-heightorientation
const media = window.matchMedia("(max-width: 850px)")

MediaQueryList对象中有两个属性:

  1. matches:返回查问后果,是一个布尔值
  2. media:媒体查问的字符串

所以,依据matches咱们就能够晓得是否命中以后的媒体查问选项

并且,咱们还能够通过addEventListener监听mediaQuery查问后果的变动

media.removeEventListener('change', () => {
  console.log(media.matches)
});

2、瀑布流组件

响应式管制

依据屏幕大小确定图片的列数

export interface PropConfigType {
  gap: number[];  // 行列的间距
  columns: number[];  // 不同媒体查问选项对应的列数
  medias?: number[];  // 媒体查问选项
}
import { reactive, onUnmounted } from 'vue';

export function useMediaValues(config: PropConfigType) {
  const info = reactive({
    columns: 1,
    gap: 1
  });

  const { columns, gap, media } = config;
  if (!media) {
    return setInfo(0);
  }

  // 循环查问medias
  const mediaQueries = media.map(media =>
    window.matchMedia(`(min-width: ${media}px)`)
  );

  const onSizeChange = () => {

    // 记录匹配到的哪一个media
    let matches = 0;
    mediaQueries.forEach(mediaQuery => {
      mediaQuery.matches && matches++;
    });

    // 优先匹配前面的
    const idx = Math.min(mediaQueries.length - 1, matches);
    setInfo(idx);
  };

  onSizeChange();

  // 监听查问后果
  for (let mediaQuery of mediaQueries) {
    mediaQuery.addEventListener('change', onSizeChange);
  }

  onUnmounted(() => {
    for (let mediaQuery of mediaQueries) {
      mediaQuery.removeEventListener('change', onSizeChange);
    }
  });

  function setInfo(idx: number) {
    info.columns = columns[idx];
    info.gap = gap[idx];
    return info;
  }

  return info;
}

数据结构

info.columns计算出瀑布流的数据结构,每列为一个数组,并存储到dataColumns

通过for循环将fall-row给渲染进去

当屏幕宽度变动时,会从新计算info.columns,失去一个新的dataColumms,渲染等到新的瀑布流界面

// fall-row组件

<div
  :style="{
    display: 'grid',
    rowGap: prop.gap,
    gridTemplateColumns: `minmax(0, 1fr)`
  }"
>
  // 用户的内容通过插槽的模式增加
  <slot></slot>
</div>
// water-fall 组件
<div
  :style="{
    display: 'grid',
    alignItems: 'start',
    gridColumnGap: info.gap + 'px',
    gridTemplateColumns: `repeat(${info.columns}, minmax(0, 1fr))`
  }"
>
  <fallRow
    v-for="(columns, index) in dataColumns"
    :key="index"
    :gap="info.gap + 'px'"
  >
    // 此处为作用域插槽
    <slot
      v-for="(src, index) in columns"
      :key="index"
      :src="src"
    ></slot>
  </fallRow>
</div>
  • 通过作用域插槽将图片的src传递给父组件

3、应用

// 导入组件
import { waterFall } from ...

<waterFall
  :data="datasource"
  :config="{
    columns: [1, 2, 3],
    gap: [24, 12, 6],
    medias: [640, 1024, 1280]
  }"
  v-slot="slotProps"
>
  <img
    :src="slotProps.src"
    :style="{ width: '100%', height: 'auto' }"
  />
</waterFall>
  • v-slot="slotProps"拜访父组件的作用域,获取到src进行显示

须要留神的是:columnsgapmedias的个数要统一

公布npm

做好工程化后,就能够通过公布为npm

地址:vue3-plock

当初能够通过依赖命令来装置了

pnpm add vue3-plock

点击codesandbox查看线上的运行

源码曾经上传github,欢送start 😀😀


参考:

  1. https://github.com/askides/react-plock
  2. https://developer.mozilla.org/zh-CN/docs/Web/API/Window/match…

评论

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理