Jetpack Compose는 디지털 인터페이스를 만들기 위한 포괄적인 디자인 시스템인 Material Design 구현을 지원합니다. Material Design 구성요소(버튼, 카드, 스위치 등)는 제품 브랜드를 효과적으로 반영하도록 Material Design을 맞춤설정하는 체계적인 방법인 머티리얼 테마 설정을 기반으로 빌드됩니다. Material 테마에는 색상, 서체 및 도형 속성이 포함되어 있습니다. 이러한 속성을 맞춤설정하면 앱을 빌드하는 데 사용되는 구성요소에 변경사항이 자동으로 반영됩니다.
Jetpack Compose는 MaterialTheme
컴포저블을 사용하여 이러한 개념을 구현합니다.
MaterialTheme( colors = // ... typography = // ... shapes = // ... ) { // app content }
MaterialTheme
에 전달하는 매개변수를 구성하여 애플리케이션의 테마를 설정합니다.
그림 1. 첫 번째 스크린샷은 MaterialTheme
을 구성하지 않는 앱을 보여주며 기본 스타일 지정을 사용합니다. 두 번째 스크린샷은 MaterialTheme
에 매개변수를 전달하여 스타일 지정을 맞춤설정하는 앱을 보여줍니다.
색상
색상은 Compose에서 간단한 데이터 보유 클래스인 Color
클래스로 모델링됩니다.
val Red = Color(0xffff0000) val Blue = Color(red = 0f, green = 0f, blue = 1f)
원하는 대로 색상을 구성할 수 있지만(예: 싱글톤 내에서 또는 정의된 인라인으로 최상위 상수로 구성) 테마에 색상을 지정하고 거기에서 색상을 검색하는 것이 좋습니다. 이 접근 방식을 사용하면 어두운 테마 및 중첩 테마를 손쉽게 지원할 수 있습니다.
그림 2. Material 색상 시스템
Compose는 Colors
클래스를 제공하여 머티리얼 색상 시스템을 모델링합니다. Colors
는 다음과 같이 밝거나 어두운 색상 세트를 만드는 빌더 함수를 제공합니다.
private val Yellow200 = Color(0xffffeb46) private val Blue200 = Color(0xff91a4fc) // ... private val DarkColors = darkColors( primary = Yellow200, secondary = Blue200, // ... ) private val LightColors = lightColors( primary = Yellow500, primaryVariant = Yellow400, secondary = Blue700, // ... )
Colors
를 정의한 후에는 MaterialTheme
에 전달할 수 있습니다.
MaterialTheme( colors = if (darkTheme) DarkColors else LightColors ) { // app content }
테마 색상 사용
MaterialTheme.colors
를 사용하여 MaterialTheme
컴포저블에 제공된 Colors
를 검색할 수 있습니다.
Text( text = "Hello theming", color = MaterialTheme.colors.primary )
표면 및 콘텐츠 색상
많은 구성요소가 한 쌍의 색상 및 콘텐츠 색상을 허용합니다.
Surface( color = MaterialTheme.colors.surface, contentColor = contentColorFor(color), // ... ) { /* ... */ } TopAppBar( backgroundColor = MaterialTheme.colors.primarySurface, contentColor = contentColorFor(backgroundColor), // ... ) { /* ... */ }
이를 통해 컴포저블의 색상을 설정할 수 있을 뿐만 아니라 그 안에 포함된 컴포저블인 콘텐츠의 기본 색상을 제공할 수도 있습니다. 많은 컴포저블은 기본적으로 이 콘텐츠 색상을 사용합니다. 예를 들어 Text
는 상위 요소의 콘텐츠 색상을 기반으로 색상을 설정하고 Icon
은 이 색상을 사용하여 색조를 설정합니다.
그림 3. 배경 색상을 다르게 설정하면 다른 텍스트 및 아이콘 색상이 생성됩니다.
contentColorFor()
메서드는 테마 색상에 적절한 '설정' 색상을 검색합니다. 예를 들어 Surface
에 primary
배경 색상을 설정하면 메서드는 이 함수를 사용하여 onPrimary
를 콘텐츠 색상으로 설정합니다. 테마가 아닌 배경 색상을 설정하는 경우 적절한 콘텐츠 색상도 지정해야 합니다. 계층 구조에 지정된 위치에서 현재 배경에 선호되는 콘텐츠 색상을 검색하려면 LocalContentColor
를 사용합니다.
콘텐츠 알파
콘텐츠를 강조하는 정도를 달리하여 중요도를 전달하고 시각적 계층 구조를 알려야 할 때가 많습니다. Material Design 텍스트 가독성 권장사항에서는 불투명도를 다르게 하여 중요도를 구분하도록 권장합니다.
Jetpack Compose에서는 LocalContentAlpha
를 사용해 이를 구현합니다.
CompositionLocal
값을 제공하여 계층 구조의 콘텐츠 알파를 지정할 수 있습니다.
중첩된 컴포저블은 이 값을 사용하여 콘텐츠에 알파 처리를 적용할 수 있습니다.
예를 들어 Text
및 Icon
은 기본적으로 LocalContentAlpha
를 사용하도록 조정된 LocalContentColor
조합을 사용합니다. 머티리얼에서는 ContentAlpha
객체에 의해 모델링된 일부 표준 알파 값(high
, medium
, disabled
)을 지정합니다.
// By default, both Icon & Text use the combination of LocalContentColor & // LocalContentAlpha. De-emphasize content by setting content alpha CompositionLocalProvider(LocalContentAlpha provides ContentAlpha.medium) { Text( // ... ) } CompositionLocalProvider(LocalContentAlpha provides ContentAlpha.disabled) { Icon( // ... ) Text( // ... ) }
CompositionLocal
에 관한 자세한 내용은 CompositionLocal을 사용한 로컬 범위 지정 데이터 가이드를 참고하세요.
그림 4. 텍스트에 다양한 수준의 강조를 적용하여 정보 계층 구조를 시각적으로 전달할 수 있습니다. 텍스트의 첫 번째 줄은 제목입니다. 가장 중요한 정보를 포함하고 있으므로 ContentAlpha.high
를 사용합니다. 두 번째 줄은 덜 중요한 메타데이터를 포함하고 있으므로 ContentAlpha.medium
을 사용합니다.
어두운 테마
Compose에서 MaterialTheme
컴포저블에 다양한 Colors
세트를 제공하여 밝은 테마 및 어두운 테마를 구현합니다.
@Composable fun MyTheme( darkTheme: Boolean = isSystemInDarkTheme(), content: @Composable () -> Unit ) { MaterialTheme( colors = if (darkTheme) DarkColors else LightColors, /*...*/ content = content ) }
이 예에서 MaterialTheme
은 어두운 테마를 사용할지 여부를 지정하는 매개변수를 받는 자체의 구성 가능한 함수로 래핑되어 있습니다. 이 경우 함수는 기기 테마 설정을 쿼리하여 darkTheme
의 기본값을 가져옵니다.
다음과 같은 코드를 사용하여 현재 Colors
가 밝은지 아니면 어두운지 확인할 수 있습니다.
val isLightTheme = MaterialTheme.colors.isLight Icon( painterResource( id = if (isLightTheme) { R.drawable.ic_sun_24 } else { R.drawable.ic_moon_24 } ), contentDescription = "Theme" )
고도 오버레이
머티리얼에서 고도가 높은 어두운 테마의 표시 경로는 배경을 밝게 하는 고도 오버레이를 수신합니다. 표시 경로의 고도가 높을수록(암시적 광원에 더 가깝게 상승) 표시 경로가 더 밝아집니다.
이러한 오버레이는 어두운 색상을 사용할 때 Surface
컴포저블에 의해 표시 경로를 사용하는 다른 Material 컴포저블에 자동으로 적용됩니다.
Surface( elevation = 2.dp, color = MaterialTheme.colors.surface, // color will be adjusted for elevation /*...*/ ) { /*...*/ }
그림 5. 카드와 하단 탐색은 모두 surface
색상을 배경으로 사용합니다. 카드와 하단 탐색은 배경에서 서로 다른 고도 수준에 있기 때문에 색상이 약간 다릅니다. 카드가 배경보다 더 밝고 하단 탐색이 카드보다 더 밝습니다.
Surface
를 사용하지 않는 맞춤 시나리오에는 Surface
구성요소에 사용되는 ElevationOverlay
가 포함된 CompositionLocal
인 LocalElevationOverlay
를 사용합니다.
// Elevation overlays // Implemented in Surface (and any components that use it) val color = MaterialTheme.colors.surface val elevation = 4.dp val overlaidColor = LocalElevationOverlay.current?.apply( color, elevation )
고도 오버레이를 중지하려면 컴포저블 계층 구조의 원하는 지점에 null
을
제공합니다.
MyTheme { CompositionLocalProvider(LocalElevationOverlay provides null) { // Content without elevation overlays } }
제한된 강조 색상
머티리얼에서는 대부분의 경우 primary
색상보다 surface
색상을 사용하는 것을 선호하기 때문에 어두운 테마에 제한된 강조 색상을 적용할 것을 권장합니다. TopAppBar
, BottomNavigation
같은 머티리얼 컴포저블에서는 기본적으로 이 동작을 구현합니다.
그림 6. 제한된 강조 색상과 함께 사용된 머티리얼 어두운 테마. 상단 앱 바는 밝은 테마에는 기본 색상을 사용하고, 어두운 테마에는 표면 경로 색상을 사용합니다.
맞춤 시나리오에는 primarySurface
확장 속성을 사용합니다.
Surface( // Switches between primary in light theme and surface in dark theme color = MaterialTheme.colors.primarySurface, /*...*/ ) { /*...*/ }
서체
머티리얼은 유형 시스템을 정의하여 의미론적으로 이름이 지정된 소수의 스타일을 사용하도록 권장합니다.
그림 7. 머티리얼 유형 시스템.
Compose는 Typography
, TextStyle
및 글꼴 관련 클래스를 사용하여 유형 시스템을 구현합니다. Typography
생성자는 각 스타일의 기본값을 제공하므로 맞춤설정하지 않으려는 스타일은 생략할 수 있습니다.
val raleway = FontFamily( Font(R.font.raleway_regular), Font(R.font.raleway_medium, FontWeight.W500), Font(R.font.raleway_semibold, FontWeight.SemiBold) ) val myTypography = Typography( h1 = TextStyle( fontFamily = raleway, fontWeight = FontWeight.W300, fontSize = 96.sp ), body1 = TextStyle( fontFamily = raleway, fontWeight = FontWeight.W600, fontSize = 16.sp ) /*...*/ ) MaterialTheme(typography = myTypography, /*...*/) { /*...*/ }
전체적으로 동일한 서체를 사용하려면 다음과 같이 defaultFontFamily parameter
를 지정하고 TextStyle
요소의 fontFamily
를 생략합니다.
val typography = Typography(defaultFontFamily = raleway) MaterialTheme(typography = typography, /*...*/) { /*...*/ }
텍스트 스타일 사용
TextStyle
에는 MaterialTheme.typography
를 통해 액세스됩니다. 다음과 같이 TextStyle
를 검색합니다.
Text( text = "Subtitle2 styled", style = MaterialTheme.typography.subtitle2 )
그림 8. 다양한 서체와 스타일을 사용하여 브랜드를 표현할 수 있습니다.
도형
Material은 도형 시스템을 정의합니다. 이 시스템을 통해 대형, 중형, 소형 구성요소의 도형을 정의할 수 있습니다.
그림 9. 머티리얼 도형 시스템.
Compose는 Shapes
클래스를 통해 도형 시스템을 구현합니다. 이 클래스를 사용하면 각 크기 카테고리의 CornerBasedShape
를 지정할 수 있습니다.
val shapes = Shapes( small = RoundedCornerShape(percent = 50), medium = RoundedCornerShape(0f), large = CutCornerShape( topStart = 16.dp, topEnd = 0.dp, bottomEnd = 0.dp, bottomStart = 16.dp ) ) MaterialTheme(shapes = shapes, /*...*/) { /*...*/ }
많은 구성요소가 기본적으로 이러한 도형을 사용합니다. 예를 들어 Button
, TextField
및 FloatingActionButton
은 기본적으로 소형으로, AlertDialog
는 기본적으로 중형으로, ModalDrawer
은 기본적으로 대형으로 설정됩니다. 전체 매핑은 도형 구성표 참조를 참고하세요.
도형 사용
Shape
에는 MaterialTheme.shapes
를 통해 액세스됩니다. 다음과 같은 코드를 사용하여 Shape
를 검색합니다.
Surface( shape = MaterialTheme.shapes.medium, /*...*/ ) { /*...*/ }
그림 10. 도형을 사용하여 브랜드 또는 상태를 표현할 수 있습니다.
기본 스타일
Compose에는 Android 보기의 기본 스타일에 상응하는 개념이 없습니다. 머티리얼 구성요소를 래핑하는 구성 가능한 '오버로드' 함수를 직접 생성하여 비슷한 기능을 제공할 수 있습니다. 예를 들어 버튼 스타일을 만들려면 고유의 구성 가능한 함수로 버튼을 래핑하고 변경하려는 매개변수를 직접 설정하며 포함하는 컴포저블에 매개변수로 다른 매개변수를 노출합니다.
@Composable fun MyButton( onClick: () -> Unit, modifier: Modifier = Modifier, content: @Composable RowScope.() -> Unit ) { Button( colors = ButtonDefaults.buttonColors( backgroundColor = MaterialTheme.colors.secondary ), onClick = onClick, modifier = modifier, content = content ) }
테마 오버레이
MaterialTheme
컴포저블을 중첩하여, Compose의 Android 보기에서 테마 오버레이와 동일한 기능을 구현할 수 있습니다. MaterialTheme
가 색상, 서체, 도형의 기본값을 현재 테마 값으로 설정하기 때문에, 테마에 이러한 매개변수 중 하나만 설정되는 경우 다른 매개변수는 기본값을 유지합니다.
더욱이 뷰 기반 화면을 Compose로 이전할 때 android:theme
속성을 사용하는 것에 유의하세요. Compose UI 트리의 관련 부분에 새로운 MaterialTheme
가 필요할 수 있습니다.
이 예에서 세부정보 화면에서는 화면 대부분에 PinkTheme
를 사용하고 관련 섹션에 BlueTheme
를 사용합니다. 아래 스크린샷과 코드를 참고하세요.
그림 11. 중첩된 테마
@Composable fun DetailsScreen(/* ... */) { PinkTheme { // other content RelatedSection() } } @Composable fun RelatedSection(/* ... */) { BlueTheme { // content } }
구성요소 상태
상호작용(클릭, 전환 등)할 수 있는 머티리얼 구성요소는 서로 다른 시각적 상태일 수 있습니다. 상태에는 사용 설정, 중지, 누름 상태 등이 있습니다.
종종 컴포저블에는 enabled
매개변수가 있습니다. 이 매개변수를 false
로 설정하면 상호작용이 방지되고, 색상과 고도 같이 구성요소 상태를 시각적으로 전달하기 위한 속성이 변경됩니다.
그림 12. enabled = true
(왼쪽) 및 enabled = false
(오른쪽) 상태의 버튼.
대부분의 경우 색상과 고도 같은 값에는 기본값을 사용할 수 있습니다. 서로 다른 상태에 사용되는 값을 구성해야 하는 경우 클래스와 편의성 함수를 사용할 수 있습니다. 아래 버튼 예시를 참고하세요.
Button( onClick = { /* ... */ }, enabled = true, // Custom colors for different states colors = ButtonDefaults.buttonColors( backgroundColor = MaterialTheme.colors.secondary, disabledBackgroundColor = MaterialTheme.colors.onBackground .copy(alpha = 0.2f) .compositeOver(MaterialTheme.colors.background) // Also contentColor and disabledContentColor ), // Custom elevation for different states elevation = ButtonDefaults.elevation( defaultElevation = 8.dp, disabledElevation = 2.dp, // Also pressedElevation ) ) { /* ... */ }
그림 13. 색상과 고도 값이 조정된, enabled = true
(왼쪽)와 enabled = false
(오른쪽) 상태의 버튼.
물결
Material 구성요소는 물결 효과를 사용하여 상호작용 중임을 나타냅니다. 계층 구조에서 MaterialTheme
를 사용하는 경우 Ripple
은 수정자 내에 clickable
, indication
같은 기본 Indication
으로 사용됩니다.
대부분의 경우 기본 Ripple
을 사용할 수 있습니다. 모양을 구성하려면 RippleTheme
을 사용하여 색상과 알파 같은 속성을 변경하면 됩니다.
RippleTheme
를 확장하고 defaultRippleColor
및 defaultRippleAlpha
유틸리티 함수를 사용할 수 있습니다. 그런 다음 LocalRippleTheme
를 사용하여 계층 구조에서 맞춤 물결 테마를 제공할 수 있습니다.
@Composable fun MyApp() { MaterialTheme { CompositionLocalProvider( LocalRippleTheme provides SecondaryRippleTheme ) { // App content } } } @Immutable private object SecondaryRippleTheme : RippleTheme { @Composable override fun defaultColor() = RippleTheme.defaultRippleColor( contentColor = MaterialTheme.colors.secondary, lightTheme = MaterialTheme.colors.isLight ) @Composable override fun rippleAlpha() = RippleTheme.defaultRippleAlpha( contentColor = MaterialTheme.colors.secondary, lightTheme = MaterialTheme.colors.isLight ) }
그림 14. RippleTheme
을 통해 제공된 서로 다른 물결 효과 값의 버튼
자세히 알아보기
Compose의 머티리얼 테마 설정에 관한 자세한 내용은 다음 추가 리소스를 참고하세요.
Codelabs
동영상
추천 서비스
- 참고: JavaScript가 사용 중지되어 있으면 링크 텍스트가 표시됩니다.
- Compose의 맞춤 디자인 시스템
- Compose에서 Material 2에서 Material 3으로 이전
- Compose의 접근성