[react] 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;
}