목록 아이템 확장 트랜지션
최근 변경일:



목록의 아이템을 클릭/터치하면 해당 아이템의 상세 정보를 확장 트랜지션으로 보여준다.
엑스칼리버: 제타착용 레벨 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 문서가 아주 일품이다. 일을 하면서 비슷한 문제를 해결해야 되는 경우가 있었는데, 그 때 나의 결과물에 비하면 추상화나 문서 수준이 너무 좋다.
참고
본문에 사용된 특정 이미지 및 명칭 대한 저작권은 SQUARE ENIX에 있습니다.
© SQUARE ENIX Published in Korea by Actoz Soft CO., LTD.