목록 아이템 확장 트랜지션

최근 변경일:

목록의 아이템을 클릭/터치하면 해당 아이템의 상세 정보를 확장 트랜지션으로 보여준다.

엑스칼리버: 제타착용 레벨 135아이템 레벨 135
아이템 레벨 135

추가 능력치

  • +41
  • 활력+48
이지스: 제타착용 레벨 135아이템 레벨 135
아이템 레벨 135

추가 능력치

  • +16
  • 활력+19
롱기누스: 제타착용 레벨 135아이템 레벨 135
아이템 레벨 135

추가 능력치

  • +57
  • 활력+67
    'use client';

    import { AnimatePresence, motion } from 'framer-motion';
    import { useRef, useState } from 'react';
    import styles from './styles.module.css';

    const reactUse = await import('react-use');
    const { useClickAway, useKey } = reactUse.default || reactUse;

    export function App() {
      const [activeItem, setActiveItem] = useState<Item | null>(null);

      useKey('Escape', () => setActiveItem(null));

      return (
        <div className={styles.listWrapper}>
          {ITEMS.map((item) => (
            <Item key={item.id} item={item} setActiveItem={setActiveItem} />
          ))}
          <AnimatePresence>
            {activeItem ? (
              <ActiveItem activeItem={activeItem} setActiveItem={setActiveItem} />
            ) : null}
          </AnimatePresence>
        </div>
      );
    }

    function Item({
      item,
      setActiveItem,
    }: { item: Item; setActiveItem: (item: Item) => void }) {
      return (
        <motion.div
          layoutId={`item-${item.id}`}
          whileTap={{ scale: 0.98 }}
          onClick={() => setActiveItem(item)}
          className={styles.item}
          tabIndex={0}
          aria-label={`${item.name} 상세정보 보기`}
          onKeyDown={(e) => e.key === 'Enter' && setActiveItem(item)}
        >
          <motion.div
            layoutId={`item-header-${item.id}`}
            className={styles.itemHeader}
          >
            <motion.img
              src={item.icon}
              width={40}
              height={40}
              layoutId={`item-icon-${item.id}`}
              className={styles.icon}
            />
            <div className={styles.headerDescription}>
              <motion.span
                layoutId={`item-name-${item.id}`}
                className={styles.itemName}
              >
                {item.name}
              </motion.span>
              <motion.span className={styles.itemHeaderLevelWrapper}>
                <span className={styles.itemHeaderLevelLabel}>
                  착용 레벨{' '}
                  <span className={styles.itemHeaderLevel}>{item.itemLevel}</span>
                </span>
                <span className={styles.itemHeaderLevelLabel}>
                  아이템 레벨{' '}
                  <span className={styles.itemHeaderLevel}>{item.itemLevel}</span>
                </span>
              </motion.span>
            </div>
            <motion.button
              aria-hidden
              tabIndex={-1}
              layoutId={`close-button-${item.id}`}
              className={styles.closeButton}
              style={{ opacity: 0 }}
            >
              <svg
                xmlns="http://www.w3.org/2000/svg"
                fill="none"
                viewBox="0 0 24 24"
                strokeWidth="2"
                height="20"
                width="20"
                stroke="currentColor"
              >
                <title>Close button</title>
                <path
                  strokeLinecap="round"
                  strokeLinejoin="round"
                  d="M6 18 18 6M6 6l12 12"
                />
              </svg>
            </motion.button>
          </motion.div>
          <motion.div
            layoutId={`item-detail-${item.id}`}
            className={styles.itemDetail}
            style={{ position: 'absolute', top: '100%', opacity: 0 }}
          >
            <motion.div className={styles.itemDetailLevel}>
              <span>아이템 레벨 {item.itemLevel}</span>
            </motion.div>
            <motion.div className={styles.baseParameterWrapper}>
              <h2 className={styles.baseParameterHeader}>추가 능력치</h2>
              <ul className={styles.baseParameterList}>
                {item.baseParam.map((param) => (
                  <li key={param.name}>
                    <span className={styles.baseParameterLabel}>{param.name}</span>+
                    {param.value}
                  </li>
                ))}
              </ul>
            </motion.div>
          </motion.div>
        </motion.div>
      );
    }

    function ActiveItem({
      activeItem,
      setActiveItem,
    }: {
      activeItem: Item;
      setActiveItem: (item: Item | null) => void;
    }) {
      const ref = useRef<HTMLDivElement>(null);
      useClickAway(ref, () => setActiveItem(null));

      return (
        <motion.div
          ref={ref}
          layoutId={`item-${activeItem.id}`}
          className={`${styles.item} ${styles.activeItem}`}
        >
          <motion.div
            layoutId={`item-header-${activeItem.id}`}
            className={styles.itemHeader}
          >
            <motion.img
              src={activeItem.icon}
              width={40}
              height={40}
              layoutId={`item-icon-${activeItem.id}`}
              className={styles.icon}
            />
            <div className={styles.headerDescription}>
              <motion.span
                layoutId={`item-name-${activeItem.id}`}
                className={styles.itemName}
              >
                {activeItem.name}
              </motion.span>
            </div>
            <motion.button
              layoutId={`close-button-${activeItem.id}`}
              className={styles.closeButton}
              aria-label="Close button"
              onClick={() => setActiveItem(null)}
            >
              <svg
                xmlns="http://www.w3.org/2000/svg"
                fill="none"
                viewBox="0 0 24 24"
                strokeWidth="2"
                height="20"
                width="20"
                stroke="currentColor"
              >
                <title>Close button</title>
                <path
                  strokeLinecap="round"
                  strokeLinejoin="round"
                  d="M6 18 18 6M6 6l12 12"
                />
              </svg>
            </motion.button>
          </motion.div>
          <motion.div layoutId={`item-detail-${activeItem.id}`}>
            <motion.div className={styles.itemDetailLevel}>
              <span>아이템 레벨 {activeItem.itemLevel}</span>
            </motion.div>
            <motion.div className={styles.baseParameterWrapper}>
              <h2 className={styles.baseParameterHeader}>추가 능력치</h2>
              <ul className={styles.baseParameterList}>
                {activeItem.baseParam.map((param) => (
                  <li key={param.name}>
                    <span className={styles.baseParameterLabel}>{param.name}</span>+
                    {param.value}
                  </li>
                ))}
              </ul>
            </motion.div>
          </motion.div>
        </motion.div>
      );
    }

    type Item = {
      id: number;
      name: string;
      icon: string;
      isUnique: boolean;
      isUntradable: boolean;
      itemLevel: number;
      itemCategoryName: string;
      physDamage: number;
      delay: number;
      baseParam: {
        name: string;
        value: number;
      }[];
    };

    const ITEMS: Item[] = [
      {
        id: 10054,
        name: '엑스칼리버: 제타',
        icon: '',
        isUnique: true,
        isUntradable: true,
        itemLevel: 135,
        itemCategoryName: '한손검',
        physDamage: 58,
        delay: 2320,
        baseParam: [
          {
            name: '힘',
            value: 41,
          },
          {
            name: '활력',
            value: 48,
          },
        ],
      },
      {
        id: 10063,
        name: '이지스: 제타',
        icon: '',
        isUnique: true,
        isUntradable: true,
        itemLevel: 135,
        itemCategoryName: '방패',
        physDamage: 0,
        delay: 0,
        baseParam: [
          {
            name: '힘',
            value: 16,
          },
          {
            name: '활력',
            value: 19,
          },
        ],
      },
      {
        id: 10057,
        name: '롱기누스: 제타',
        icon: '',
        isUnique: true,
        isUntradable: true,
        itemLevel: 135,
        itemCategoryName: '양손창',
        physDamage: 58,
        delay: 2960,
        baseParam: [
          {
            name: '힘',
            value: 57,
          },
          {
            name: '활력',
            value: 67,
          },
        ],
      },
    ];

이런 류의 효과를 구현할 때 매우 손이 많이 가는 편인데, motion(구 framer-motion)을 사용하여 비교적 간편하게 구현할 수 있다.

참고로 motion의 일부 내부 구현 원리를 다룬 Inside Framer’s Magic Motion 문서가 아주 일품이다. 일을 하면서 비슷한 문제를 해결해야 되는 경우가 있었는데, 그 때 나의 결과물에 비하면 추상화나 문서 수준이 너무 좋다.

참고