Skip to content

๐ŸŽง์Œ์•…์„ ํƒ์ƒ‰ํ•˜๊ณ  ์žฌ์ƒํ•  ์ˆ˜ ์žˆ๋Š” ์›น ๋ฎค์ง ํ”Œ๋ ˆ์ด์–ด

Notifications You must be signed in to change notification settings

SuDaJo/MusicPlayer

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

Repository files navigation

๐ŸŽตย Music Drop

๐Ÿ”— ๋ฐฐํฌURL


ํ”„๋กœ์ ํŠธ ์†Œ๊ฐœ

[ ๊ฐœ์š” ]

  • Music Drop์€ ์Œ์•…์„ ๋“ค์„ ์ˆ˜ ์žˆ๋Š” ์›น ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์ž…๋‹ˆ๋‹ค.
  • open API (deezer)๋ฅผ ํ™œ์šฉํ•˜์—ฌ ๋‹ค์–‘ํ•œ ์Œ์•…์„ ๊ฒ€์ƒ‰ ๋ฐ ์ถ”๊ฐ€ ํ•  ์ˆ˜ ์žˆ๊ณ  30์ดˆ ๋ฏธ๋ฆฌ๋“ฃ๊ธฐ๋ฅผ ์ œ๊ณตํ•ฉ๋‹ˆ๋‹ค.
  • ๊ตญ๋‚ด, ํ•ด์™ธ์ฐจํŠธ ์ˆœ์œ„๋ฅผ ํ™•์ธํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.
  • ๋‚˜๋งŒ์˜ ํ”Œ๋ ˆ์ด๋ฆฌ์ŠคํŠธ๋ฅผ ๋งŒ๋“ค์–ด ์›ํ•˜๋Š” ์Œ์•…์„ ๋“ค์„ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.
  • ํ”Œ๋ ˆ์ด๋ฆฌ์ŠคํŠธ์— ๋‹ด๊ธด ์Œ์•…์„ ๊ธฐ๋ฐ˜์œผ๋กœ ๋žœ๋ค ์žฌ์ƒ, ํ•œ๊ณก ๋ฐ˜๋ณต, ๋‹ค์Œ๊ณก/์ด์ „๊ณก ์žฌ์ƒ ๊ธฐ๋Šฅ์„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.
  • ๋ชจ๋“  ๊ธฐ๋Šฅ๊ตฌํ˜„์„ VSC live share๋ฅผ ํ™œ์šฉํ•˜์—ฌ ํŽ˜์–ด ํ”„๋กœ๊ทธ๋ž˜๋ฐ์œผ๋กœ ์ง„ํ–‰ํ–ˆ์Šต๋‹ˆ๋‹ค.

[ ์ œ์ž‘ ๊ธฐ๊ฐ„ ]

2023.01.19 - 2023.03.31

โฌ†๏ธTop

๐Ÿ“ ๋ชฉ์ฐจ

1. ํ”„๋กœ์ ํŠธ ์†Œ๊ฐœ
2. ํŒ€์› ์†Œ๊ฐœ
3. ๊ฐœ๋ฐœ ํ™˜๊ฒฝ
4. ๊ตฌํ˜„ ๊ธฐ๋Šฅ
5. ํ”„๋กœ์ ํŠธ ๊ตฌ์กฐ
6. ํŠธ๋Ÿฌ๋ธ” ์ŠˆํŒ…
7. ํ”„๋กœ์ ํŠธ ์ง„ํ–‰ ์ค‘ ์ƒˆ๋กœ ์•Œ๊ฒŒ ๋œ ๊ฒƒ๋“ค
8. ํ”„๋กœ์ ํŠธ ๋ฐฐํฌ ํ›„ ๋А๋‚€ ์ 


๐Ÿ™‹โ€โ™€๏ธ ํŒ€์› ์†Œ๊ฐœ

๊น€๋‹ค์ • ์ •์ˆ˜ํ˜„ ์กฐ๋ฏผ์ง€
DaJeong_profile_img SuHyun_profile_img MinJi_profile_img
DADA6041 IntHyun ming-Jo

โฌ†๏ธTop

โš™๏ธ ๊ฐœ๋ฐœ ํ™˜๊ฒฝ

[ ๊ธฐ์ˆ  ์Šคํƒ ]

[ ํ˜•์ƒ๊ด€๋ฆฌ ๋ฐ ํ˜‘์—… ๋„๊ตฌ ]

  • GitHub Issues : ํ™œ๋™ ๋‚ด์—ญ์— ๋Œ€ํ•ด ์ด์Šˆ ๋“ฑ๋ก ํ›„ ์ž‘์—… ์ง„ํ–‰
  • VScode liveshare : ํŽ˜์–ด ํ”„๋กœ๊ทธ๋ž˜๋ฐ์œผ๋กœ ์ง„ํ–‰ ๋ฐ ์ฝ”๋“œ ๋ฆฌ๋ทฐ
  • Discord : ์‹ค์‹œ๊ฐ„ ์†Œํ†ต ๋ฐ ํ™”๋ฉด ๊ณต์œ 

[ ์ด์Šˆ ๊ด€๋ฆฌ ]

  • ํ™œ๋™ ๋‚ด์—ญ์— ๋Œ€ํ•ด ์ด์Šˆ ๋“ฑ๋ก ํ›„ ์ž‘์—… ์ง„ํ–‰
    issues

  • GitHub Project ์‚ฌ์šฉ
    Project

  • GitHub ์ฝ”๋“œ๋ฆฌ๋ทฐ
    review

[ ์ปค๋ฐ‹ ์ปจ๋ฒค์…˜ ]

โœจ Feat: ์ƒˆ๋กœ์šด ๊ธฐ๋Šฅ ์ถ”๊ฐ€
๐Ÿท๏ธ MarkUp: ๋งˆํฌ์—…
๐Ÿ’„ Design : CSS, ์‚ฌ์šฉ์ž UI ๋””์ž์ธ ๋ณ€๊ฒฝ
๐Ÿ“ Docs : ๋ฌธ์„œ ์ˆ˜์ •
๐ŸŽจ Style : ์ฝ”๋“œํฌ๋งท, ์„ธ๋ฏธ์ฝœ๋ก , ๊ฐœํ–‰, ์ฝ”๋“œ ๊ตฌ์กฐ, ํ˜•ํƒœ
๐Ÿค” Test : ํ…Œ์ŠคํŠธ ์ฝ”๋“œ
โ™ป๏ธ Refactor: ์ฝ”๋“œ ๋ฆฌํŒฉํ† ๋ง
๐Ÿ› Fix : ๋ฒ„๊ทธ ๋ฐ ์˜ค๋ฅ˜ ์ˆ˜์ •
๐Ÿ”ฅ Remove : ๋ถˆํ•„์š”ํ•œ ํŒŒ์ผ ์‚ญ์ œ
๐Ÿ”จ Chore : ๋นŒ๋“œ ์—…๋ฌด, ํŒจํ‚ค์ง€๋งค๋‹ˆ์ €, ํด๋”ํŠธ๋ฆฌ, ์„ธํŒ… ์ˆ˜์ •
๐Ÿ’ก Comment : ํ•„์š”ํ•œ ์ฃผ์„ ์ถ”๊ฐ€ ๋ฐ ๋ณ€๊ฒฝ

Co-authored-by: DADA6061 <d.ya96v@gmail.com>
Co-authored-by: ming-Jo <mwinterj@daum.net>
Co-authored-by: IntHyun <wjdtngus1237@naver.com>

โฌ†๏ธTop

๐Ÿ›  ๊ตฌํ˜„ ๊ธฐ๋Šฅ

Splash Home Chart
Playlist Search(๊ธฐ๋ณธ) Search(๊ฒฐ๊ณผ ์—†์Œ)
Music Control Not Found

โฌ†๏ธTop

๐Ÿ—‚ ํ”„๋กœ์ ํŠธ ๊ตฌ์กฐ

๐Ÿ“ฆfrontend
โ”ฃ ๐Ÿ“‚static
โ”ƒ โ”ฃ ๐Ÿ“‚css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œchart.css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œhome.css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œmyplaylist.css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œnavbar.css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œplaycontrol.css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œreset.css
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œsearch.css
โ”ƒ โ”ƒ โ”— ๐Ÿ“œstyle.css
โ”ƒ โ”ฃ ๐Ÿ“‚image
โ”ƒ โ”— ๐Ÿ“‚js
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“‚API
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œAPI.js
โ”ƒ โ”ƒ โ”ƒ โ”— ๐Ÿ“œapikey.js
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“‚json
โ”ƒ โ”ƒ โ”ƒ โ”— ๐Ÿ“œmemberData.json
โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“‚views
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œAbstractView.js
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œChart.js
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œHome.js
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œMyplayList.js
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œNotFound.js
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œPlayControl.js
โ”ƒ โ”ƒ โ”ƒ โ”ฃ ๐Ÿ“œSearch.js
โ”ƒ โ”ƒ โ”ƒ โ”— ๐Ÿ“œSplash.js
โ”ƒ โ”ƒ โ”— ๐Ÿ“œindex.js
โ”ฃ ๐Ÿ“œindex.html
โ”— ๐Ÿ“œ_redirects

โฌ†๏ธTop

๐Ÿ™‡โ€โ™€๏ธ ํŠธ๋Ÿฌ๋ธ” ์ŠˆํŒ…

1. ๋‹ค๋ฅธ ํŽ˜์ด์ง€๋กœ ์ด๋™ํ•  ๋•Œ ๋งˆ๋‹ค ์š”์†Œ๊ฐ€ ์ถ”๊ฐ€๋จ

  • ๋ฌธ์ œ ๋ฐ ์›์ธ : createElement๋กœ ์š”์†Œ๋ฅผ ์ƒ์„ฑํ•˜๊ณ  appendChild๋กœ ์š”์†Œ๋ฅผ ์ถ”๊ฐ€ํ•˜๊ณ  ์žˆ์–ด์„œ ๋ฉ”์„œ๋“œ๊ฐ€ ์‹คํ–‰๋  ๋•Œ ๋งˆ๋‹ค ์š”์†Œ๊ฐ€ ์ƒ์„ฑ, ์ถ”๊ฐ€๋จ

  • ํ•ด๊ฒฐ :

    • (์ฐธ๊ณ ) ๊ฐ ํŽ˜์ด์ง€์—๋Š” root ์š”์†Œ ์•ˆ์— main ์ปจํ…Œ์ด๋„ˆ ์š”์†Œ๊ฐ€ ๊ณตํ†ต์ ์œผ๋กœ ์กด์žฌํ•จ

    • 1์ฐจ ํ•ด๊ฒฐ

      • ํด๋ฆญํ•œ ํŽ˜์ด์ง€์˜ main wrapper์— ์š”์†Œ๋ฅผ hiddenํ•˜๋Š” class๋ฅผ ์ถ”๊ฐ€

      • ๋ผ์šฐํŒ… ํ›„ hidden๋œ ์š”์†Œ๊ฐ€ DOM์— ์กด์žฌํ•˜์ง€ ์•Š์•„ ์—๋Ÿฌ ๋ฐœ์ƒ

    • 2์ฐจ ํ•ด๊ฒฐ

      • removeElement() ํ•จ์ˆ˜๋ฅผ ์ƒ์„ฑํ•˜์—ฌ ํŽ˜์ด์ง€ ์ด๋™ ์‹œ ์ง์ „ ํŽ˜์ด์ง€์˜ ์š”์†Œ๋ฅผ ์ œ๊ฑฐํ•˜๋Š” ๋กœ์ง ๊ตฌํ˜„

      • ๋’ค๋กœ๊ฐ€๊ธฐ ๋ฒ„ํŠผ ํด๋ฆญ ์‹œ ์ œ๊ฑฐ๋˜์—ˆ๋˜ ์š”์†Œ๊ฐ€ ๋‹ค์‹œ ์ถ”๊ฐ€๋˜๋Š” ํ˜„์ƒ ๋ฐœ์ƒ

    • 3์ฐจ ํ•ด๊ฒฐ(์™„๋ฃŒ)

      • ๊ฐ ํŽ˜์ด์ง€์˜ main wrapper๋ฅผ appendChild๊ฐ€ ์•„๋‹Œ replaceChildren์œผ๋กœ ํŽ˜์ด์ง€ ์ด๋™ ์‹œ ์š”์†Œ ๋Œ€์ฒด

      • replaceChildren MDN ์„ค๋ช… ์ฐธ๊ณ 

        // ๊ธฐ์กด appendChild
        const wrapper = document.createElement("main");
        wrapper.classList.add("wrapper");
        this.$target.appendChild(wrapper);
        
        // ์ˆ˜์ • replaceChildren
        const wrapper = document.createElement("main");
        wrapper.classList.add("wrapper");
        this.$target.replaceChildren(wrapper);

2. ์žฌ์ƒ๋ฐ” ๋“œ๋ž˜๊ทธ ์‹œ ์ •์ƒ์ ์œผ๋กœ ๋“œ๋ž˜๊ทธ ์œ„์น˜ ๋ชป ์žก๋Š” ํ˜„์ƒ

  • ๋ฌธ์ œ ๋ฐ ์›์ธ : ์Œ์•…์„ ์žฌ์ƒํ•  ๋•Œ ์žฌ์ƒ๋ฐ”๋ฅผ ๋“œ๋ž˜๊ทธ ํ•˜๋ฉด ๋“œ๋ž˜๊ทธํ•œ ์œ„์น˜๋ฅผ ์ •์ƒ์ ์œผ๋กœ ํ‘œ์‹œํ•˜์ง€ ์•Š์Œ

  • ํ•ด๊ฒฐ : input์˜ range๋Š” click์ด ์•„๋‹ˆ๋ผ input event๋กœ ์ปจํŠธ๋กค ํ•ด์•ผํ•œ๋‹ค.

    // click ์ด๋ฒคํŠธ๋กœ ํ–ˆ์„ ๋•Œ (๋“œ๋ž˜๊ทธ ์œ„์น˜ ํŒŒ์•… X)
    $progressBar.addEventListener("click", (e) => {
      let progressInputValue = e.target.value;
      let songDuration = $audio.duration;
      $audio.currentTime = (progressInputValue * songDuration) / 100;
    });
    
    // input ์ด๋ฒคํŠธ๋กœ ํ–ˆ์„ ๋•Œ (๋“œ๋ž˜๊ทธ ์œ„์น˜ ํŒŒ์•… O)
    $progressBar.addEventListener("input", (e) => {
      let progressInputValue = e.target.value;
      let songDuration = $audio.duration;
      $audio.currentTime = (progressInputValue * songDuration) / 100;
    });

3. ํ™”์‚ดํ‘œ ํ•จ์ˆ˜ ๋‚ด๋ถ€์—์„œ ๋ฉ”์„œ๋“œ ์‹คํ–‰ ์•ˆ๋˜๋Š” ํ˜„์ƒ

  • ๋ฌธ์ œ ๋ฐ ์›์ธ : ํ™”์‚ดํ‘œ ํ•จ์ˆ˜ ๋‚ด๋ถ€์—์„œ this๊ฐ€ ์ง€์นญํ•˜๋Š” ๋Œ€์ƒ์ด ๋‹ฌ๋ผ์„œ ๋ถ€๋ชจ Class์˜ ๋ฉ”์„œ๋“œ๊ฐ€ ์ž‘๋™ํ•˜์ง€ ์•Š์Œ

  • ํ•ด๊ฒฐ : this.method()๊ฐ€ ์•„๋‹Œ super.method()๋กœ ๋ถ€๋ชจ Class์˜ prototype-method๋ฅผ ์ฐธ์กฐํ•˜๊ธฐ

      setLocalData() {
        const btn = document.querySelectorAll(".music-add-btn");
        btn.forEach((button) => {
          button.addEventListener("click", (event) => {
            super.toast();
    
            let id = Number(event.currentTarget.dataset.id);
            let title = event.currentTarget.dataset.title;
            let coverImg = event.currentTarget.dataset.cover;
            let artist = event.currentTarget.dataset.artist;
    
            // ๊ธฐ์กด์— ์ž‘์„ฑํ–ˆ๋˜ this๋Š” ์ƒ์œ„ ํด๋ž˜์Šค์˜ setLocalStorage()๋ฅผ ์ฐพ์ง€๋ชปํ•จ.
            // this.setLocalStorage(id, title, coverImg, artist);
    
            super.setLocalStorage(id, title, coverImg, artist);
          });
        });
      }

4. ๋น„๋™๊ธฐ๋กœ DOM ์š”์†Œ ์ƒ์„ฑ ์‹œ event ์ž‘๋™ ์•ˆํ•จ

  • ๋ฌธ์ œ : anchor ํƒœ๊ทธ์˜ ์ƒˆ๋กœ๊ณ ์นจ์„ ๋ฐฉ์ง€ํ•˜๋ฉฐ ํŽ˜์ด์ง€๋ฅผ ์ด๋™ํ•˜๊ธฐ ์œ„ํ•ด anchor ํƒœ๊ทธ์— ์ ‘๊ทผํ•˜๋Š” ๊ณผ์ •์—์„œ anchor ํƒœ๊ทธ๋ฅผ ์ฐพ์ง€ ๋ชปํ•˜๋Š” ๋ฌธ์ œ ๋ฐœ์ƒ

  • ์›์ธ : createElement๋กœ ๋ชจ๋“  ์š”์†Œ๋“ค์„ ์ƒ์„ฑํ•˜์˜€๋Š”๋ฐ ๋น„๋™๊ธฐ ํ†ต์‹ ์œผ๋กœ ์ธํ•ด ์š”์†Œ๊ฐ€ ์ƒ์„ฑ๋˜๊ธฐ ์ „ DOM์— ์ ‘๊ทผํ•˜์—ฌ ํ•ด๋‹น ์š”์†Œ๋ฅผ ์ฐพ์ง€ ๋ชปํ•จ

  • ํ•ด๊ฒฐ : ๋น„๋™๊ธฐ ํ†ต์‹  ์ดํ›„ ๋ณ€๊ฒฝ๋œ DOM ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ์ง€ํ•˜๊ธฐ ์œ„ํ•ด MutationObserver๋กœ ์†์„ฑ๊ณผ ์ž์‹๋…ธ๋“œ, ์ž์†๋…ธ๋“œ์— ๋ณ€๊ฒฝ๋˜๋Š” ์š”์†Œ๊ฐ€ ์žˆ๋Š”์ง€ ๊ฐ์ง€ํ•œ ๋’ค ํ•ด๋‹น ์š”์†Œ์— ์ ‘๊ทผํ•˜์—ฌ event๋ฅผ ์ ์šฉ

    let target;
    
    if (location.pathname === "/") {
      target = document.querySelector(".home-wrapper");
    } else if (location.pathname === "/search") {
      target = document.querySelector(".searchlist-main");
    } else if (location.pathname === "/chart") {
      target = document.querySelector(".chart-main");
    }
    
    if (target) {
      const callback = () => {
        const $playControl = target.querySelectorAll(".to-play-control");
        $playControl.forEach((url) => {
          url.addEventListener("click", (e) => {
            e.preventDefault();
            navigateTo(e.currentTarget.href);
          });
        });
      };
    
      const observer = new MutationObserver(callback);
    
      const config = {
        attributes: true, // ์†์„ฑ ๋ณ€ํ™” ํ• ๋•Œ ๊ฐ์ง€
        childList: true, // ์ž์‹๋…ธ๋“œ ์ถ”๊ฐ€/์ œ๊ฑฐ ๊ฐ์ง€
        subtree: true, // ์†์ž๋…ธ๋“œ๊นŒ์ง€ ์ถ”๊ฐ€/์ œ๊ฑฐ ๊ฐ์ง€
        characterData: true, // ๋ฐ์ดํ„ฐ ๋ณ€๊ฒฝ์ „ ๋‚ด์šฉ ๊ธฐ๋ก
      };
    
      observer.observe(target, config);
    }

โฌ†๏ธTop


โœ๏ธ ํ”„๋กœ์ ํŠธ ์ง„ํ–‰ ์ค‘ ์ƒˆ๋กœ ์•Œ๊ฒŒ ๋œ ๊ฒƒ๋“ค

  • Promise ๊ฐ์ฒด๋ฅผ ์ƒ์„ฑํ•˜์—ฌ ๋ถˆ๋Ÿฌ์˜จ ๋ฐ์ดํ„ฐ๋ฅผ ๋ Œ๋”๋ง ํ•˜๊ธฐ ์ „ ๋กœ๋”ฉํ™”๋ฉด ๋„์šฐ๊ธฐ

    new Promise((resolve) => {
      const contCategory = data
        .map((list, idx) => {
          return `
            ... ํŽ˜์ด์ง€ ์š”์†Œ ์ƒ์„ฑํ•˜๋Š” ์ฝ”๋“œ ...
            `;
        })
        .join("");
    
      resolve(/* ์ƒ์„ฑ๋˜๊ณ  ๋‚œ ๋’ค ๋„˜๊ฒจ์ค„ ๋ฐ์ดํ„ฐ */);
    })
      .then((/* resolve์—์„œ ๋ฐ›์€ ๋ฐ์ดํ„ฐ */) => {
        // ๋ฐ์ดํ„ฐ๋ฅผ ๋ถˆ๋Ÿฌ์˜ค๊ณ  ๋‚œ ๋’ค ์‹คํ–‰ํ•  ์ฝ”๋“œ
      })
      .catch((error) => {
        throw new Error(error);
      });
  • scroll method ํŠน์ • ์œ„์น˜๋กœ ์Šคํฌ๋กค ํ•˜๊ธฐ

    • scrollBy MDN ์„ค๋ช… ์ฐธ๊ณ 

    • scrollBy๋Š” left, top, behavior ํ”„๋กœํผํ‹ฐ๋ฅผ ๊ฐ€์ง

    • left๋Š” x์ถ• / top์€ y์ถ• ์ด๋™

    • behavior

      • auto(๊ธฐ๋ณธ๊ฐ’) : ๋ธŒ๋ผ์šฐ์ €๊ฐ€ ์„ ํƒ
      • smooth : ๋ถ€๋“œ๋Ÿฌ์šด ์Šคํฌ๋กค ์• ๋‹ˆ๋ฉ”์ด์…˜
      • instant : ํ•œ ๋ฒˆ์˜ ์ ํ”„๋กœ ์ฆ‰์‹œ ์‹คํ–‰
    • Home ํ™”๋ฉด ๊ฐ€๋กœ ์Šคํฌ๋กค ๋ฒ„ํŠผ ๊ตฌํ˜„ ์‹œ ์‚ฌ์šฉ

    horizontalScroll() {
      const $leftBtns = document.querySelectorAll(".left-scroll-button");
      const $rightBtns = document.querySelectorAll(".right-scroll-button");
    
      $leftBtns.forEach((item) => {
        item.addEventListener("click", () => {
          const contAlbum = item.previousElementSibling;
          if (contAlbum.scrollLeft + contAlbum.offsetWidth >= contAlbum.scrollWidth) {
    
            contAlbum.scrollBy({ left: -290, top: 0, behavior: "smooth" });
          } else {
            contAlbum.scrollBy({ left: -260, top: 0, behavior: "smooth" });
          }
        });
      });
    
      $rightBtns.forEach((item) => {
        item.addEventListener("click", () => {
          const contAlbum = item.previousElementSibling.previousElementSibling;
          contAlbum.scrollBy({ left: 260, top: 0, behavior: "smooth" });
        });
      });
    }
  • css filter ์†์„ฑ์œผ๋กœ asset ์ถ”๊ฐ€ ์—†์ด ์ด๋ฏธ์ง€ ์ปฌ๋Ÿฌ ๋ณ€๊ฒฝํ•˜๊ธฐ

    • ๋‚ด๋น„๊ฒŒ์ด์…˜ ์•„์ด์ฝ˜ active ํšจ๊ณผ ๊ตฌํ˜„
    .nav-list-item.active {
      filter: invert(63%) sepia(51%) saturate(3249%) hue-rotate(136deg) brightness(93%) contrast(97%);
    }
    const menuItems = document.querySelectorAll(".nav-list-item");
    
    let currentActive;
    
    menuItems.forEach((item) => {
      const itemHref = item.querySelector("a").getAttribute("href");
      if (location.pathname === "/") {
        if (itemHref === "/") {
          currentActive = item;
          currentActive.classList.add("active");
        }
      } else if (location.pathname.includes(itemHref) && itemHref.length !== 1) {
        currentActive = item;
        currentActive.classList.add("active");
      }
    });

โฌ†๏ธTop


๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง ํ”„๋กœ์ ํŠธ ๋ฐฐํฌ ํ›„ ๋А๋‚€ ์ 

๐Ÿผ ๊น€๋‹ค์ •

  • ๋ฉ‹์‚ฌ ๊ณผ์ •์ด ๋๋‚˜๊ณ  ์ผ๋‹จ ์ด๋ ฅ์„œ์— ์“ธ ๊ฒฐ๊ณผ๋ฌผ์„ ๋” ๋งŒ๋“ค์–ด์•ผ ํ•œ๋‹ค๋Š” ์ƒ๊ฐ์— ์‹œ์ž‘ํ•œ ํ”„๋กœ์ ํŠธ์ง€๋งŒ ๊ธฐ์ˆ ์ ์ธ ๊ฒƒ๋ฟ ์•„๋‹ˆ๋ผ ์†Œํ†ตํ•˜๋Š” ๋ฐฉ๋ฒ•๊นŒ์ง€ ๋„ˆ๋ฌด ๋งŽ์€ ๊ฒƒ์„ ๋ฐฐ์šฐ๊ณ  ์–ป์–ด ๊ฐ€๋Š” ์‹œ๊ฐ„์ด์—ˆ์Šต๋‹ˆ๋‹ค. ํ”„๋กœ์ ํŠธ๋ฅผ ์ง„ํ–‰ํ•˜๋ฉด์„œ ์ฒซ ํŽ˜์–ด ํ”„๋กœ๊ทธ๋ž˜๋ฐ์— ์ดˆ๊ธฐ ์„ธํŒ… ๋“ฑ ๋งค๋ฒˆ ๊ณ ๋น„๊ฐ€ ์™”์ง€๋งŒ ์–ด๋А์ƒˆ ์ž๋ž‘ํ•˜๊ณ  ์‹ถ์€ ์ €์˜ ๋ฉ”์ธ ํ”„๋กœ์ ํŠธ์ž…๋‹ˆ๋‹ค. ์–ด๋ ค์šด ๋ถ€๋ถ„์ด ๋‚˜์˜ฌ ๋•Œ๋งˆ๋‹ค ๋ชจ๋‘ ์ ๊ทน์ ์œผ๋กœ ๋‚˜์„œ๊ณ  ๊ฐ™์ด ํ•ด๊ฒฐํ•˜๋ ค๊ณ  ๋…ธ๋ ฅํ•œ ๋•์— ์ˆœํƒ„ํ•˜๊ฒŒ ์ž˜ ๋งˆ๋ฌด๋ฆฌํ•˜๊ฒŒ ๋œ ๊ฒƒ ๊ฐ™๊ณ  ์•ž์œผ๋กœ๋„ ๋” ์—ด์‹ฌํžˆ ํ•ด์•ผ๊ฒ ๋‹ค๋Š” ์›๋™๋ ฅ๋„ ์–ป์—ˆ๊ณ  ์ž์‹ ๊ฐ๋„ ์–ป์–ด ๊ฐ€๊ฒŒ ๋˜์—ˆ์Šต๋‹ˆ๋‹ค. :) ์ •ํ•ด์ง„ ๋งˆ๊ฐ๊ธฐํ•œ์ด ์žˆ์ง€๋„ ์•Š์€ ๋ฐ๋‹ค ํ•˜๋ฃจ 2์‹œ๊ฐ„, 2๋‹ฌ์ด๋ž€ ๊ธฐ๊ฐ„์ด ๊ธธ๋‹ค๋ฉด ๊ธธ๊ณ  ์งง์œผ๋ฉด ์งง๋‹ค๊ณ  ๋А๋ผ๊ฒ ์ง€๋งŒ ํฌ๊ธฐํ•˜์ง€ ์•Š๊ณ  ๊พธ์ค€ํžˆ ๊ฐ™์ด ์ง„ํ–‰ํ•ด ์ค€ ํŒ€์›๋“ค ๊ฐ์‚ฌํ•˜๊ณ  ๋„ˆ๋ฌด ๊ณ ์ƒ ๋งŽ์œผ์…จ์Šต๋‹ˆ๋‹ค!โค๏ธ

๐Ÿ’— ์ •์ˆ˜ํ˜„

  • ํ”„๋กœ์ ํŠธ๋ฅผ ์‹œ์ž‘ํ•˜๊ธฐ ์ „์—๋Š” ๊ฐ์ฒด์ง€ํ–ฅ์— ๋Œ€ํ•ด ๋ชจ๋ฅด๋Š” ๊ฒƒ์ด ๋งŽ์€ ์ƒํƒœ์˜€๋Š”๋ฐ Class๋ฌธ๋ฒ•์„ ์‹ค์ œ๋กœ ์‚ฌ์šฉํ•ด๋ณด๊ณ  ํŠน์ • ๋ฉ”์„œ๋“œ๋ฅผ ์ƒ์œ„ Class์— ์ ์šฉํ•ด ์žฌํ™œ์šฉ๋„ ํ•ด๋ณด๋ฉด์„œ Class๋ฌธ๋ฒ•์ด ์ข€ ๋” ์ต์ˆ™ํ•ด์ง„ ๊ฒƒ ๊ฐ™์Šต๋‹ˆ๋‹ค. ๋˜ํ•œ ์ด์ „์—๋Š” ์•Œ์ง€ ๋ชปํ–ˆ๋˜ css์˜ filter์†์„ฑ, scrollBy ๋ฉ”์„œ๋“œ, ๋™์ ์œผ๋กœ DOM์„ ๊ฐ์ง€ํ•  ์ˆ˜ ์žˆ๋Š” MutationObserver ๋“ฑ์— ๋Œ€ํ•ด ๋ฐฐ์› ๊ณ  netlify๋ฅผ ํ†ตํ•ด ์ง์ ‘ ๋ฐฐํฌ๋ฅผ ํ•˜๊ณ  github Releasese๋กœ ๋ฒ„์ „๊ด€๋ฆฌ๋„ ํ•ด๋ณด๋ฉด์„œ ์›น ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์ด ์ด๋Ÿฐ์‹์œผ๋กœ ๋ฒ„์ „๊ด€๋ฆฌ๊ฐ€ ๋˜๊ณ  ๋ฐฐํฌ๊ฐ€ ๋˜๋Š”๊ตฌ๋‚˜์— ๋Œ€ํ•ด ์ข€ ๋” ์ž์„ธํ•˜๊ฒŒ ์•Œ๊ฒŒ ๋œ ๊ฒƒ ๊ฐ™์Šต๋‹ˆ๋‹ค. ์ด๋ฒˆ ํ”„๋กœ์ ํŠธ๋ฅผ ํ†ตํ•ด JavaScript์— ๋Œ€ํ•ด ๋” ๋งŽ์ด ์•Œ๊ฒŒ ๋˜์—ˆ๊ณ  ์ดํ›„์—๋„ ๋‹ค์–‘ํ•œ ํ”„๋กœ์ ํŠธ๋ฅผ ํ†ตํ•ด ์ง€์‹์„ ํ™•์žฅํ•ด๋‚˜๊ฐ€๊ณ  ์‹ถ๋‹ค๋Š” ์ƒ๊ฐ์ด ๋“ค์—ˆ์Šต๋‹ˆ๋‹ค.

๐Ÿ˜Ž ์กฐ๋ฏผ์ง€

  • ๋ฐ”๋‹๋ผ ์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ๋กœ ๊ตฌํ˜„ํ•˜๋Š” ๊ณผ์ •์ด ์ƒ๊ฐ๋ณด๋‹ค ์‰ฝ์ง€ ์•Š์•˜์ง€๋งŒ ๋‹ค์ •๋‹˜, ์ˆ˜ํ˜„๋‹˜๊ณผ ํ•จ๊ป˜์—ฌ์„œ ์ฒ˜์Œ์— ์ƒ๊ฐํ–ˆ๋˜ ๊ฒƒ๋ณด๋‹ค ๋” ๋ฉ‹์ง€๊ฒŒ ํ”„๋กœ์ ํŠธ๋ฅผ ์™„์„ฑํ•  ์ˆ˜ ์žˆ์—ˆ์Šต๋‹ˆ๋‹ค! ๊ตฌํ˜„ํ•˜๋ฉด์„œ ๋ง‰ํžˆ๋Š” ๋ถ€๋ถ„์ด ์žˆ์„ ๋•Œ '์ผ๋‹จ ๋ญ๋“  ํ•ด๋ณผ๊นŒ์š”?' ๋ผ๋Š” ๋งˆ์ธ๋“œ๋กœ ๋‹ค๊ฐ™์ด ์šฐ๋‹นํƒ•ํƒ• ํ•ด๋‚˜๊ฐ€๋Š” ๊ณผ์ •์ด ๋„ˆ๋ฌด ์žฌ๋ฐŒ์—ˆ์–ด์š”๐Ÿ˜ ํ”„๋กœ์ ํŠธ๋ฅผ ์ง„ํ–‰ํ•˜๋ฉด์„œ ํ‰์†Œ์— ์–ด๋ ต๋‹ค๊ณ  ์ƒ๊ฐํ–ˆ๋˜ ํด๋ž˜์Šค ๋ฌธ๋ฒ•, ๋น„๋™๊ธฐ ํŒจํ„ด๊ณผ ๋งˆ์ฃผํ•˜๊ฒŒ ๋˜์–ด ๊ฑฑ์ •์ด ๋งŽ์•˜์ง€๋งŒ ํ•จ๊ป˜ ์ •๋ฉด๋ŒํŒŒ๋กœ ํ•ด๋‚ด๊ฒŒ ๋˜์–ด ๋„ˆ๋ฌด ๋ฟŒ๋“ฏํ•ฉ๋‹ˆ๋‹ค๐Ÿ˜Ž ์•ž์œผ๋กœ ๋” ๋ฉ‹์ง„ ๊ฐœ๋ฐœ์ž๊ฐ€ ๋  ์ˆ˜๋‹ค์กฐ ์‚ผ๋‘๋งˆ์ฐจ ์—ฌ๋Ÿฌ๋ถ„ ์ •๋ง ์ˆ˜๊ณ ํ•˜์…จ์Šต๋‹ˆ๋‹ค๐Ÿ’ช

โฌ†๏ธTop

About

๐ŸŽง์Œ์•…์„ ํƒ์ƒ‰ํ•˜๊ณ  ์žฌ์ƒํ•  ์ˆ˜ ์žˆ๋Š” ์›น ๋ฎค์ง ํ”Œ๋ ˆ์ด์–ด

Resources

Stars

Watchers

Forks

Packages

No packages published

Contributors 3

  •  
  •  
  •