LearnApplyShare

[react] dark 모드 구현

June 09, 2020 - [react, dark-mode, dark]

요즘 유행하는 다크모드를 리액트에서 구현하는 방법입니다. 기본적인 컨셉은 설정된 테마에 따라 body 태그의 클래스를 다르게 변경하는 것입니다. 기본적인 컨셉만 이해하면 누구든지 자유롭게 자신의 사이트 상황에 맞게 적절히 적용할 수 있을 것이라 생각합니다. 아래 방법은 이 블로그에서 사용 중인 다크모드의 구현사례입니다.

1. dark / light 모드에서 사용할 색상 정의

우선 각 테마 별로 사용할 색상을 css 변수를 이용해 전역으로 정의합니다. 그러면 각 컴포넌트에서는 color: var(--textNormal); 과 같이 해당 변수에 접근할 수 있습니다.

/* global.scss */
body {
  background-color: var(--bg);
  &.light {
    --bg: #ffffff;
    --textTitle: #444;
    --textNormal: #555;
    --textDesc: #999;
  }
  &.dark {
    --bg: #282c35;
    --textTitle: #eee;
    --textNormal: #ddd;
    --textDesc: #888;
  }
}

2. theme context 정의

테마의 설정 값을 애플리케이션 전역의 상태값으로 사용하기 위해서 context api 를 이용합니다. 그리고 설정한 테마값을 새로고침시에도 유지하기 위해 localStorage 를 사용합니다. (아래 예제는 theme 키에 dark or light 값을 사용합니다. 선호에 따라 dark 라는 키값에 true, false 값을 사용하기도 합니다)

// ThemeContext.js
// https://www.gatsbyjs.org/blog/2019-01-31-using-react-context-api-with-gatsby/
import React from 'react'

const defaultState = {
  theme: null,
  setTheme: () => {},
}

const ThemeContext = React.createContext(defaultState)

// Getting dark mode information from OS!
// You need macOS Mojave + Safari Technology Preview Release 68 to test this currently.
// const supportsDarkMode = () =>
//   window.matchMedia("(prefers-color-scheme: dark)").matches === true

export class ThemeProvider extends React.Component {
  state = {
    theme: null,
  }

  setTheme = theme => {
    document.body.className = theme    /* 모바일 브라우져에서 theme-color 를 함께 변경하고자 할 경우, 아래 2줄 주석해제 */
    // const meta = document.querySelector('meta[name="theme-color"]')
    // meta.content = theme === 'light' ? '#eee' : '#282c35'
    localStorage.setItem('theme', theme)
    this.setState({theme})
  }

  componentDidMount() {
    this.setTheme(localStorage.getItem('theme') || 'light')
  }

  render() {
    const {children} = this.props
    const {theme} = this.state
    return (
      <ThemeContext.Provider
        value={{
          theme,
          setTheme: this.setTheme,
        }}
      >
        {children}
      </ThemeContext.Provider>
    )
  }
}

export default ThemeContext

ThemeContext 사용을 위해 애플리케이션 전체를 ThemeProvider 컴포넌트로 wrapping

// App.js
import React from 'react'
import {ThemeProvider} from './themeContext'

export default () => (
  <ThemeProvider>
    <App />
  </ThemeProvider>
)

3. 입력컨트롤 정의

이제 dark / light 모드를 제어할 입력 컨트롤이 필요합니다. 이래와 같이 단순하게 Dark, Light 텍스트만 출력하는 형태로 만들어도 동작에는 문제가 없습니다. DarkMode 컴포넌트는 ThemeContext 를 사용해야 하기 때문에 ThemeContext.Consumer 로 wrapping 됩니다

// DarkMode.js
import React from 'react'
import ThemeContext from './themeContext'

export default function DarkMode() {
  return (
    <ThemeContext.Consumer>
      {ctx => (
        <div onClick={() => ctx.setTheme(ctx.theme === 'light' ? 'dark' : 'light')}>
          {ctx.theme}
        </div>
      )}
    </ThemeContext.Consumer>
  )
}

이제 완성된 DarkMode 컴포넌트를 원하는 위치에 삽입하기만 하면 됩니다. 😁🙂


4. 멋진 입력컨트롤 사용

애정을 가지고 운영하는 사이트라면 위의 제시된 DarkMode 컴포넌트를 그대로 사용하지는 않겠죠? 다크모드 컨트롤의 형태는 여러가지가 있는데요. 일단 현 블로그에서 사용 중인 다크모드 컨트롤의 구현은 아래와 같습니다.

Note) 아래 다크모드 컨트롤은 overreacted.io 에서 사용 중인 컨트롤을 그대로 가져온 것입니다.

// DarkMode.js
import React from 'react'
import ThemeContext from './themeContext'
import Toggle from './toggle'
import {moon, sun} from './icon'

export default function DarkMode() {
  return (
    <ThemeContext.Consumer>
      {ctx =>
        ctx.theme && ( // ctx.theme 가 null 일 경우에는 렌더링하지 않음(깜빡인 문제 해결)
          <Toggle
            icons={{
              checked: (
                <img
                  src={moon}
                  alt='moon'
                  width='16'
                  height='16'
                  role='presentation'
                  style={{pointerEvents: 'none'}}
                />
              ),
              unchecked: (
                <img
                  src={sun}
                  alt='sun'
                  width='16'
                  height='16'
                  role='presentation'
                  style={{pointerEvents: 'none'}}
                />
              ),
            }}
            checked={ctx.theme === 'dark'}
            onChange={e => ctx.setTheme(e.target.checked ? 'dark' : 'light')}
          />
        )
      }
    </ThemeContext.Consumer>
  )
}
// icon.js
export const moon = ``
export const sun = ``
// toggle.js
/*
 * Copyright (c) 2015 instructure-react
 * Forked from https://github.com/aaronshaf/react-toggle/
 * + applied https://github.com/aaronshaf/react-toggle/pull/90
 **/

import './toggle.scss'

import React, {PureComponent} from 'react'

// Copyright 2015-present Drifty Co.
// http://drifty.com/
// from: https://github.com/driftyco/ionic/blob/master/src/util/dom.ts
function pointerCoord(event) {
  // get coordinates for either a mouse click
  // or a touch depending on the given event
  if (event) {
    const changedTouches = event.changedTouches
    if (changedTouches && changedTouches.length > 0) {
      const touch = changedTouches[0]
      return {x: touch.clientX, y: touch.clientY}
    }
    const pageX = event.pageX
    if (pageX !== undefined) {
      return {x: pageX, y: event.pageY}
    }
  }
  return {x: 0, y: 0}
}

export default class Toggle extends PureComponent {
  constructor(props) {
    super(props)
    this.handleClick = this.handleClick.bind(this)
    this.handleTouchStart = this.handleTouchStart.bind(this)
    this.handleTouchMove = this.handleTouchMove.bind(this)
    this.handleTouchEnd = this.handleTouchEnd.bind(this)
    this.handleTouchCancel = this.handleTouchCancel.bind(this)
    this.handleFocus = this.handleFocus.bind(this)
    this.handleBlur = this.handleBlur.bind(this)
    this.previouslyChecked = !!(props.checked || props.defaultChecked)
    this.state = {
      checked: !!(props.checked || props.defaultChecked),
      hasFocus: false,
    }
  }

  componentWillReceiveProps(nextProps) {
    if ('checked' in nextProps) {
      this.setState({checked: !!nextProps.checked})
      this.previouslyChecked = !!nextProps.checked
    }
  }

  handleClick(event) {
    const checkbox = this.input
    this.previouslyChecked = checkbox.checked
    if (event.target !== checkbox && !this.moved) {
      event.preventDefault()
      checkbox.focus()
      checkbox.click()
      return
    }

    this.setState({checked: checkbox.checked})
  }

  handleTouchStart(event) {
    this.startX = pointerCoord(event).x
    this.touchStarted = true
    this.hadFocusAtTouchStart = this.state.hasFocus
    this.setState({hasFocus: true})
  }

  handleTouchMove(event) {
    if (!this.touchStarted) return
    this.touchMoved = true

    if (this.startX != null) {
      let currentX = pointerCoord(event).x
      if (this.state.checked && currentX + 15 < this.startX) {
        this.setState({checked: false})
        this.startX = currentX
      } else if (!this.state.checked && currentX - 15 > this.startX) {
        this.setState({checked: true})
        this.startX = currentX
      }
    }
  }

  handleTouchEnd(event) {
    if (!this.touchMoved) return
    const checkbox = this.input
    event.preventDefault()

    if (this.startX != null) {
      if (this.previouslyChecked !== this.state.checked) {
        checkbox.click()
      }

      this.touchStarted = false
      this.startX = null
      this.touchMoved = false
    }

    if (!this.hadFocusAtTouchStart) {
      this.setState({hasFocus: false})
    }
  }

  handleTouchCancel(event) {
    if (this.startX != null) {
      this.touchStarted = false
      this.startX = null
      this.touchMoved = false
    }

    if (!this.hadFocusAtTouchStart) {
      this.setState({hasFocus: false})
    }
  }

  handleFocus(event) {
    const {onFocus} = this.props

    if (onFocus) {
      onFocus(event)
    }

    this.hadFocusAtTouchStart = true
    this.setState({hasFocus: true})
  }

  handleBlur(event) {
    const {onBlur} = this.props

    if (onBlur) {
      onBlur(event)
    }

    this.hadFocusAtTouchStart = false
    this.setState({hasFocus: false})
  }

  getIcon(type) {
    const {icons} = this.props
    if (!icons) {
      return null
    }
    return icons[type] === undefined ? Toggle.defaultProps.icons[type] : icons[type]
  }

  render() {
    const {className, icons: _icons, ...inputProps} = this.props
    const classes =
      'react-toggle' +
      (this.state.checked ? ' react-toggle--checked' : '') +
      (this.state.hasFocus ? ' react-toggle--focus' : '') +
      (this.props.disabled ? ' react-toggle--disabled' : '') +
      (className ? ' ' + className : '')
    return (
      <div
        className={classes}
        onClick={this.handleClick}
        onKeyPress={this.handleClick}
        onTouchStart={this.handleTouchStart}
        onTouchMove={this.handleTouchMove}
        onTouchEnd={this.handleTouchEnd}
        onTouchCancel={this.handleTouchCancel}
        role='presentation'
      >
        <div className='react-toggle-track'>
          <div className='react-toggle-track-check'>{this.getIcon('checked')}</div>
          <div className='react-toggle-track-x'>{this.getIcon('unchecked')}</div>
        </div>
        <div className='react-toggle-thumb' />

        <input
          {...inputProps}
          ref={ref => {
            this.input = ref
          }}
          onFocus={this.handleFocus}
          onBlur={this.handleBlur}
          className='react-toggle-screenreader-only'
          type='checkbox'
          aria-label='Switch between Dark and Light mode'
        />
      </div>
    )
  }
}
// toggle.scss
/*
 * Copyright (c) 2015 instructure-react
 * Forked from https://github.com/aaronshaf/react-toggle/
**/

.react-toggle {
  touch-action: pan-x;

  display: inline-block;
  position: relative;
  cursor: pointer;
  background-color: transparent;
  border: 0;
  padding: 0;

  -webkit-touch-callout: none;
  user-select: none;

  -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
  -webkit-tap-highlight-color: transparent;
}

.react-toggle-screenreader-only {
  border: 0;
  clip: rect(0 0 0 0);
  height: 1px;
  margin: -1px;
  overflow: hidden;
  padding: 0;
  position: absolute;
  width: 1px;
}

.react-toggle-track {
  width: 50px;
  height: 24px;
  padding: 0;
  border-radius: 30px;
  background-color: hsl(222, 14%, 7%);
  transition: all 0.2s ease;
}

.react-toggle-track-check {
  position: absolute;
  width: 17px;
  height: 17px;
  left: 5px;
  top: 0px;
  bottom: 0px;
  margin-top: auto;
  margin-bottom: auto;
  line-height: 0;
  opacity: 0;
  transition: opacity 0.25s ease;
}

.react-toggle--checked .react-toggle-track-check {
  opacity: 1;
  transition: opacity 0.25s ease;
}

.react-toggle-track-x {
  position: absolute;
  width: 17px;
  height: 17px;
  right: 5px;
  top: 0px;
  bottom: 0px;
  margin-top: auto;
  margin-bottom: auto;
  line-height: 0;
  opacity: 1;
  transition: opacity 0.25s ease;
}

.react-toggle--checked .react-toggle-track-x {
  opacity: 0;
}

.react-toggle-thumb {
  position: absolute;
  top: 1px;
  left: 1px;
  width: 22px;
  height: 22px;
  border-radius: 50%;
  background-color: #fafafa;
  box-sizing: border-box;
  transition: all 0.5s cubic-bezier(0.23, 1, 0.32, 1) 0ms;
  transform: translateX(0);
}

.react-toggle--checked .react-toggle-thumb {
  transform: translateX(26px);
  border-color: #19ab27;
}

.react-toggle--focus .react-toggle-thumb {
  box-shadow: 0px 0px 2px 3px #777;
}

.react-toggle:active .react-toggle-thumb {
  box-shadow: 0px 0px 5px 5px #777;
}