feat: 설정 페이지 탭 구조화 및 EdgeToEdge 개선 (v1.8.0)
- 설정 페이지를 3개 탭으로 구분 (알림, 사이트, 기타) - 탭 간 스와이프 이동 지원 - One UI 7+ EdgeToEdge 문제 해결 (시스템 바 침범 수정) - 파싱 데이터 삭제 기능 추가 - 업데이트 확인 시 즉시 다운로드 및 설치 기능 구현
This commit is contained in:
@@ -24,9 +24,8 @@ android {
|
||||
applicationId = "com.hotdeal.alarm"
|
||||
minSdk = 31
|
||||
targetSdk = 35
|
||||
versionCode = 13
|
||||
versionName = "1.7.0"
|
||||
versionName = "1.6.0"
|
||||
versionCode = 14
|
||||
versionName = "1.8.0"
|
||||
|
||||
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
|
||||
vectorDrawables {
|
||||
|
||||
@@ -2,6 +2,9 @@ package com.hotdeal.alarm.presentation.main
|
||||
|
||||
import android.content.BroadcastReceiver
|
||||
import android.os.Bundle
|
||||
import android.view.View
|
||||
import android.view.WindowInsets
|
||||
import android.view.WindowManager
|
||||
import android.widget.Toast
|
||||
import androidx.activity.ComponentActivity
|
||||
import androidx.activity.compose.setContent
|
||||
@@ -11,6 +14,9 @@ import androidx.compose.material3.*
|
||||
import androidx.compose.runtime.*
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.unit.dp
|
||||
import androidx.core.view.ViewCompat
|
||||
import androidx.core.view.WindowCompat
|
||||
import androidx.core.view.WindowInsetsCompat
|
||||
import androidx.hilt.navigation.compose.hiltViewModel
|
||||
import androidx.lifecycle.lifecycleScope
|
||||
import com.hotdeal.alarm.ui.theme.HotDealTheme
|
||||
@@ -32,9 +38,11 @@ class MainActivity : ComponentActivity() {
|
||||
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
super.onCreate(savedInstanceState)
|
||||
enableEdgeToEdge()
|
||||
|
||||
// 2분 간격으로 폴<> 시작
|
||||
// EdgeToEdge 설정 (One UI 7+ 대응)
|
||||
setupEdgeToEdge()
|
||||
|
||||
// 2분 간격으로 폴링 시작
|
||||
workerScheduler.schedulePeriodicPolling(2)
|
||||
|
||||
setContent {
|
||||
@@ -121,6 +129,61 @@ class MainActivity : ComponentActivity() {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* EdgeToEdge 설정 - One UI 7+ 대응
|
||||
*/
|
||||
private fun setupEdgeToEdge() {
|
||||
// WindowCompat을 사용한 설정
|
||||
WindowCompat.setDecorFitsSystemWindows(window, false)
|
||||
|
||||
// statusBarColor와 navigationBarColor를 투명하게
|
||||
window.statusBarColor = android.graphics.Color.TRANSPARENT
|
||||
window.navigationBarColor = android.graphics.Color.TRANSPARENT
|
||||
|
||||
// One UI 7+에서 시스템 바가 콘텐츠를 가리지 않도록 설정
|
||||
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.R) {
|
||||
window.setDecorFitsSystemWindows(false)
|
||||
|
||||
// 시스템 바 인셋 처리
|
||||
ViewCompat.setOnApplyWindowInsetsListener(window.decorView.rootView) { view, insets ->
|
||||
val systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars())
|
||||
val ime = insets.getInsets(WindowInsetsCompat.Type.ime())
|
||||
|
||||
view.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom)
|
||||
insets
|
||||
}
|
||||
} else {
|
||||
@Suppress("DEPRECATION")
|
||||
window.setFlags(
|
||||
WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS,
|
||||
WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
|
||||
)
|
||||
}
|
||||
|
||||
// 시스템 바 아이콘 색상 설정 (라이트/다크 모드)
|
||||
val decorView = window.decorView
|
||||
@Suppress("DEPRECATION")
|
||||
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.R) {
|
||||
window.insetsController?.let { controller ->
|
||||
// 상태바 아이콘 밝게 (다크 모드용)
|
||||
controller.setSystemBarsAppearance(
|
||||
0,
|
||||
android.view.WindowInsetsController.APPEARANCE_LIGHT_STATUS_BARS
|
||||
)
|
||||
// 네비게이션바 아이콘 밝게 (다크 모드용)
|
||||
controller.setSystemBarsAppearance(
|
||||
0,
|
||||
android.view.WindowInsetsController.APPEARANCE_LIGHT_NAVIGATION_BARS
|
||||
)
|
||||
}
|
||||
} else {
|
||||
@Suppress("DEPRECATION")
|
||||
decorView.systemUiVisibility = decorView.systemUiVisibility or
|
||||
View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR or
|
||||
View.SYSTEM_UI_FLAG_LIGHT_NAVIGATION_BAR
|
||||
}
|
||||
}
|
||||
|
||||
override fun onDestroy() {
|
||||
super.onDestroy()
|
||||
// 리시버 해제
|
||||
|
||||
@@ -5,21 +5,25 @@ import android.content.pm.PackageManager
|
||||
import android.os.Build
|
||||
import androidx.activity.compose.rememberLauncherForActivityResult
|
||||
import androidx.activity.result.contract.ActivityResultContracts
|
||||
import androidx.compose.foundation.ExperimentalFoundationApi
|
||||
import androidx.compose.foundation.layout.*
|
||||
import androidx.compose.foundation.pager.HorizontalPager
|
||||
import androidx.compose.foundation.pager.rememberPagerState
|
||||
import androidx.compose.material3.*
|
||||
import androidx.compose.runtime.*
|
||||
import androidx.compose.runtime.snapshotFlow
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.platform.LocalContext
|
||||
import androidx.core.content.ContextCompat
|
||||
import androidx.lifecycle.compose.collectAsStateWithLifecycle
|
||||
import androidx.navigation.NavHostController
|
||||
import androidx.navigation.compose.NavHost
|
||||
import androidx.navigation.compose.composable
|
||||
import androidx.navigation.compose.rememberNavController
|
||||
import kotlinx.coroutines.launch
|
||||
import com.hotdeal.alarm.presentation.components.PermissionDialog
|
||||
import com.hotdeal.alarm.presentation.deallist.DealListScreen
|
||||
import com.hotdeal.alarm.presentation.settings.SettingsScreen
|
||||
|
||||
@OptIn(ExperimentalFoundationApi::class)
|
||||
@Composable
|
||||
fun MainScreen(
|
||||
viewModel: MainViewModel,
|
||||
@@ -49,21 +53,43 @@ fun MainScreen(
|
||||
}
|
||||
}
|
||||
|
||||
// TopAppBar 제거 - 화면을 넓게 사용
|
||||
NavHost(
|
||||
navController = navController,
|
||||
startDestination = Screen.DealList.route
|
||||
val pagerState = rememberPagerState(
|
||||
initialPage = 0,
|
||||
pageCount = { 2 }
|
||||
)
|
||||
val coroutineScope = rememberCoroutineScope()
|
||||
var currentPage by remember { mutableStateOf(0) }
|
||||
|
||||
LaunchedEffect(pagerState) {
|
||||
snapshotFlow { pagerState.currentPage }.collect { page ->
|
||||
currentPage = page
|
||||
}
|
||||
}
|
||||
|
||||
Column(
|
||||
modifier = Modifier
|
||||
.fillMaxSize()
|
||||
.windowInsetsPadding(WindowInsets.safeDrawing) // 시스템 바 패딩 적용
|
||||
) {
|
||||
composable(Screen.DealList.route) {
|
||||
HorizontalPager(
|
||||
state = pagerState,
|
||||
modifier = Modifier.weight(1f)
|
||||
) { page ->
|
||||
when (page) {
|
||||
0 -> {
|
||||
DealListScreen(
|
||||
viewModel = viewModel,
|
||||
onNavigateToSettings = { navController.navigate(Screen.Settings.route) }
|
||||
onNavigateToSettings = {
|
||||
coroutineScope.launch { pagerState.animateScrollToPage(1) }
|
||||
}
|
||||
)
|
||||
}
|
||||
composable(Screen.Settings.route) {
|
||||
1 -> {
|
||||
SettingsScreen(viewModel = viewModel)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (showPermissionDialog) {
|
||||
PermissionDialog(
|
||||
|
||||
@@ -1,22 +1,27 @@
|
||||
package com.hotdeal.alarm.presentation.settings
|
||||
|
||||
import android.Manifest
|
||||
import android.app.DownloadManager
|
||||
import android.content.Context
|
||||
import android.content.Intent
|
||||
import android.net.Uri
|
||||
import android.os.Build
|
||||
import android.os.Environment
|
||||
import android.provider.Settings
|
||||
import android.widget.Toast
|
||||
import androidx.activity.compose.rememberLauncherForActivityResult
|
||||
import androidx.activity.result.contract.ActivityResultContracts
|
||||
import androidx.compose.animation.*
|
||||
import androidx.compose.animation.core.*
|
||||
import androidx.compose.foundation.ExperimentalFoundationApi
|
||||
import androidx.compose.foundation.background
|
||||
import androidx.compose.foundation.border
|
||||
import androidx.compose.foundation.interaction.MutableInteractionSource
|
||||
import androidx.compose.foundation.interaction.collectIsPressedAsState
|
||||
import androidx.compose.foundation.layout.*
|
||||
import androidx.compose.foundation.lazy.LazyColumn
|
||||
import androidx.compose.foundation.lazy.items
|
||||
import androidx.compose.foundation.pager.HorizontalPager
|
||||
import androidx.compose.foundation.pager.rememberPagerState
|
||||
import androidx.compose.foundation.shape.CircleShape
|
||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
||||
import androidx.compose.material.icons.Icons
|
||||
@@ -26,14 +31,11 @@ import androidx.compose.material3.*
|
||||
import androidx.compose.runtime.*
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.draw.clip
|
||||
import androidx.compose.ui.draw.scale
|
||||
import androidx.compose.ui.graphics.Brush
|
||||
import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.graphics.vector.ImageVector
|
||||
import androidx.compose.ui.platform.LocalContext
|
||||
import androidx.compose.ui.text.font.FontWeight
|
||||
import androidx.compose.ui.text.style.TextAlign
|
||||
import androidx.compose.ui.unit.dp
|
||||
import androidx.lifecycle.compose.collectAsStateWithLifecycle
|
||||
import com.hotdeal.alarm.domain.model.Keyword
|
||||
@@ -47,7 +49,7 @@ import com.hotdeal.alarm.util.PermissionHelper
|
||||
import com.hotdeal.alarm.util.VersionManager
|
||||
import kotlinx.coroutines.launch
|
||||
|
||||
@OptIn(ExperimentalMaterial3Api::class)
|
||||
@OptIn(ExperimentalFoundationApi::class, ExperimentalMaterial3Api::class)
|
||||
@Composable
|
||||
fun SettingsScreen(viewModel: MainViewModel) {
|
||||
val context = LocalContext.current
|
||||
@@ -55,6 +57,10 @@ fun SettingsScreen(viewModel: MainViewModel) {
|
||||
val scope = rememberCoroutineScope()
|
||||
val currentPollingInterval by viewModel.pollingInterval.collectAsState()
|
||||
|
||||
// 탭 상태
|
||||
val pagerState = rememberPagerState(initialPage = 0, pageCount = { 3 })
|
||||
val tabTitles = listOf("알림", "사이트", "기타")
|
||||
|
||||
var showPermissionDialog by remember { mutableStateOf(false) }
|
||||
var permissionDialogTitle by remember { mutableStateOf("") }
|
||||
var permissionDialogMessage by remember { mutableStateOf("") }
|
||||
@@ -74,6 +80,74 @@ fun SettingsScreen(viewModel: MainViewModel) {
|
||||
val permissionStatus = PermissionHelper.checkAllPermissions(context)
|
||||
val hasEnabledSites = (uiState as? MainUiState.Success)?.siteConfigs?.any { it.isEnabled } ?: false
|
||||
|
||||
Column(modifier = Modifier.fillMaxSize()) {
|
||||
// 탭 레이블
|
||||
PrimaryTabRow(
|
||||
selectedTabIndex = pagerState.currentPage,
|
||||
containerColor = MaterialTheme.colorScheme.surface,
|
||||
contentColor = MaterialTheme.colorScheme.primary
|
||||
) {
|
||||
tabTitles.forEachIndexed { index, title ->
|
||||
Tab(
|
||||
selected = pagerState.currentPage == index,
|
||||
onClick = { scope.launch { pagerState.animateScrollToPage(index) } },
|
||||
text = {
|
||||
Text(
|
||||
text = title,
|
||||
fontWeight = if (pagerState.currentPage == index) FontWeight.Bold else FontWeight.Normal
|
||||
)
|
||||
},
|
||||
selectedContentColor = MaterialTheme.colorScheme.primary,
|
||||
unselectedContentColor = MaterialTheme.colorScheme.onSurfaceVariant
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
// 탭 내용 (스와이프 가능)
|
||||
HorizontalPager(
|
||||
state = pagerState,
|
||||
modifier = Modifier.fillMaxSize(),
|
||||
beyondBoundsPageCount = 1
|
||||
) { page ->
|
||||
when (page) {
|
||||
0 -> NotificationTab(
|
||||
viewModel = viewModel,
|
||||
permissionStatus = permissionStatus,
|
||||
hasEnabledSites = hasEnabledSites,
|
||||
currentPollingInterval = currentPollingInterval,
|
||||
onRequestPermission = {
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
||||
notificationPermissionLauncher.launch(Manifest.permission.POST_NOTIFICATIONS)
|
||||
}
|
||||
}
|
||||
)
|
||||
1 -> SitesTab(viewModel = viewModel)
|
||||
2 -> MoreTab(viewModel = viewModel)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (showPermissionDialog) {
|
||||
PermissionDialog(
|
||||
title = permissionDialogTitle,
|
||||
message = permissionDialogMessage,
|
||||
onDismiss = { showPermissionDialog = false },
|
||||
onOpenSettings = { showPermissionDialog = false; permissionDialogAction() }
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun NotificationTab(
|
||||
viewModel: MainViewModel,
|
||||
permissionStatus: PermissionHelper.PermissionStatus,
|
||||
hasEnabledSites: Boolean,
|
||||
currentPollingInterval: Int,
|
||||
onRequestPermission: () -> Unit
|
||||
) {
|
||||
val context = LocalContext.current
|
||||
val uiState by viewModel.uiState.collectAsStateWithLifecycle()
|
||||
|
||||
LazyColumn(
|
||||
modifier = Modifier
|
||||
.fillMaxSize()
|
||||
@@ -81,22 +155,12 @@ fun SettingsScreen(viewModel: MainViewModel) {
|
||||
verticalArrangement = Arrangement.spacedBy(12.dp),
|
||||
contentPadding = PaddingValues(vertical = 16.dp)
|
||||
) {
|
||||
// 상단 알림 설정 섹션
|
||||
// 알림 권한 설정
|
||||
item {
|
||||
NotificationSettingsHeader(
|
||||
permissionStatus = permissionStatus,
|
||||
hasEnabledSites = hasEnabledSites,
|
||||
onRequestPermission = {
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
||||
notificationPermissionLauncher.launch(Manifest.permission.POST_NOTIFICATIONS)
|
||||
}
|
||||
},
|
||||
onOpenSystemSettings = {
|
||||
val intent = Intent(Settings.ACTION_APP_NOTIFICATION_SETTINGS).apply {
|
||||
putExtra(Settings.EXTRA_APP_PACKAGE, context.packageName)
|
||||
}
|
||||
context.startActivity(intent)
|
||||
}
|
||||
onRequestPermission = onRequestPermission
|
||||
)
|
||||
}
|
||||
|
||||
@@ -112,7 +176,48 @@ fun SettingsScreen(viewModel: MainViewModel) {
|
||||
)
|
||||
}
|
||||
|
||||
// 사이트 설정 섹션
|
||||
// 키워드 설정
|
||||
item {
|
||||
SectionHeader(
|
||||
title = "키워드 알림",
|
||||
icon = Icons.Outlined.NotificationsActive,
|
||||
description = "키워드를 등록하면 해당 키워드가 포함된 핫딜 알림"
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
KeywordInputCard(onAdd = { keyword -> viewModel.addKeyword(keyword) })
|
||||
}
|
||||
|
||||
if (uiState is MainUiState.Success) {
|
||||
items((uiState as MainUiState.Success).keywords, key = { it.id }) { keyword ->
|
||||
EnhancedKeywordCard(
|
||||
keyword = keyword,
|
||||
onToggle = { viewModel.toggleKeyword(keyword.id, !keyword.isEnabled) },
|
||||
onDelete = { viewModel.deleteKeyword(keyword.id) }
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
// 하단 여백
|
||||
item {
|
||||
Spacer(modifier = Modifier.height(32.dp))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// ==================== 사이트 탭 ====================
|
||||
@Composable
|
||||
private fun SitesTab(viewModel: MainViewModel) {
|
||||
val uiState by viewModel.uiState.collectAsStateWithLifecycle()
|
||||
|
||||
LazyColumn(
|
||||
modifier = Modifier
|
||||
.fillMaxSize()
|
||||
.padding(horizontal = 16.dp),
|
||||
verticalArrangement = Arrangement.spacedBy(12.dp),
|
||||
contentPadding = PaddingValues(vertical = 16.dp)
|
||||
) {
|
||||
item {
|
||||
SectionHeader(
|
||||
title = "사이트 선택",
|
||||
@@ -132,51 +237,6 @@ fun SettingsScreen(viewModel: MainViewModel) {
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
// 키워드 설정 섹션
|
||||
item {
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
SectionHeader(
|
||||
title = "키워드 알림",
|
||||
icon = Icons.Outlined.NotificationsActive,
|
||||
description = "키워드를 등록하면 해당 키워드가 포함된 핫딜 알림"
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
KeywordInputCard(
|
||||
onAdd = { keyword ->
|
||||
viewModel.addKeyword(keyword)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
items(state.keywords, key = { it.id }) { keyword ->
|
||||
EnhancedKeywordCard(
|
||||
keyword = keyword,
|
||||
onToggle = { viewModel.toggleKeyword(keyword.id, !keyword.isEnabled) },
|
||||
onDelete = { viewModel.deleteKeyword(keyword.id) }
|
||||
)
|
||||
}
|
||||
|
||||
// 버전 정보
|
||||
item {
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
VersionCard(
|
||||
currentVersion = VersionManager.getCurrentVersion(context),
|
||||
onCheckUpdate = {
|
||||
scope.launch {
|
||||
val currentCode = VersionManager.getCurrentVersionCode(context)
|
||||
val remoteInfo = VersionManager.checkForUpdate()
|
||||
if (remoteInfo != null && VersionManager.isUpdateAvailable(currentCode, remoteInfo.versionCode)) {
|
||||
Toast.makeText(context, "새 버전 ${remoteInfo.version} 사용 가능", Toast.LENGTH_LONG).show()
|
||||
} else {
|
||||
Toast.makeText(context, "최신 버전입니다", Toast.LENGTH_SHORT).show()
|
||||
}
|
||||
}
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
else -> {
|
||||
item {
|
||||
@@ -191,24 +251,252 @@ fun SettingsScreen(viewModel: MainViewModel) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 하단 여백
|
||||
item {
|
||||
Spacer(modifier = Modifier.height(32.dp))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// ==================== 기타 탭 ====================
|
||||
@Composable
|
||||
private fun MoreTab(viewModel: MainViewModel) {
|
||||
val context = LocalContext.current
|
||||
val scope = rememberCoroutineScope()
|
||||
var showDeleteDialog by remember { mutableStateOf(false) }
|
||||
var isCheckingUpdate by remember { mutableStateOf(false) }
|
||||
|
||||
val toastEvent by viewModel.toastEvent.collectAsStateWithLifecycle(initialValue = null)
|
||||
|
||||
LaunchedEffect(toastEvent) {
|
||||
toastEvent?.let { message ->
|
||||
Toast.makeText(context, message, Toast.LENGTH_SHORT).show()
|
||||
}
|
||||
}
|
||||
|
||||
if (showPermissionDialog) {
|
||||
PermissionDialog(
|
||||
title = permissionDialogTitle,
|
||||
message = permissionDialogMessage,
|
||||
onDismiss = { showPermissionDialog = false },
|
||||
onOpenSettings = { showPermissionDialog = false; permissionDialogAction() }
|
||||
LazyColumn(
|
||||
modifier = Modifier
|
||||
.fillMaxSize()
|
||||
.padding(horizontal = 16.dp),
|
||||
verticalArrangement = Arrangement.spacedBy(12.dp),
|
||||
contentPadding = PaddingValues(vertical = 16.dp)
|
||||
) {
|
||||
// 데이터 관리
|
||||
item {
|
||||
SectionHeader(
|
||||
title = "데이터 관리",
|
||||
icon = Icons.Outlined.Storage,
|
||||
description = "저장된 데이터를 관리합니다"
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
Card(
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
shape = RoundedCornerShape(16.dp),
|
||||
colors = CardDefaults.cardColors(
|
||||
containerColor = MaterialTheme.colorScheme.surface
|
||||
),
|
||||
elevation = CardDefaults.cardElevation(defaultElevation = 1.dp)
|
||||
) {
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(16.dp),
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(44.dp)
|
||||
.background(
|
||||
MaterialTheme.colorScheme.errorContainer,
|
||||
CircleShape
|
||||
),
|
||||
contentAlignment = Alignment.Center
|
||||
) {
|
||||
Icon(
|
||||
imageVector = Icons.Filled.DeleteSweep,
|
||||
contentDescription = null,
|
||||
tint = MaterialTheme.colorScheme.onErrorContainer,
|
||||
modifier = Modifier.size(24.dp)
|
||||
)
|
||||
}
|
||||
Spacer(modifier = Modifier.width(12.dp))
|
||||
Column(modifier = Modifier.weight(1f)) {
|
||||
Text(
|
||||
text = "파싱 데이터 삭제",
|
||||
style = MaterialTheme.typography.titleMedium,
|
||||
fontWeight = FontWeight.Bold
|
||||
)
|
||||
Text(
|
||||
text = "저장된 모든 핫딜 데이터를 삭제합니다",
|
||||
style = MaterialTheme.typography.bodySmall,
|
||||
color = MaterialTheme.colorScheme.onSurfaceVariant
|
||||
)
|
||||
}
|
||||
FilledTonalIconButton(
|
||||
onClick = { showDeleteDialog = true },
|
||||
colors = IconButtonDefaults.filledTonalIconButtonColors(
|
||||
containerColor = MaterialTheme.colorScheme.errorContainer
|
||||
)
|
||||
) {
|
||||
Icon(
|
||||
imageVector = Icons.Outlined.Delete,
|
||||
contentDescription = "삭제",
|
||||
tint = MaterialTheme.colorScheme.error
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 앱 정보
|
||||
item {
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
SectionHeader(
|
||||
title = "앱 정보",
|
||||
icon = Icons.Outlined.Info,
|
||||
description = "앱 버전 및 업데이트"
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
Card(
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
shape = RoundedCornerShape(16.dp),
|
||||
colors = CardDefaults.cardColors(
|
||||
containerColor = MaterialTheme.colorScheme.surface
|
||||
),
|
||||
elevation = CardDefaults.cardElevation(defaultElevation = 1.dp)
|
||||
) {
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(16.dp),
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(44.dp)
|
||||
.background(
|
||||
MaterialTheme.colorScheme.tertiaryContainer,
|
||||
CircleShape
|
||||
),
|
||||
contentAlignment = Alignment.Center
|
||||
) {
|
||||
Icon(
|
||||
imageVector = Icons.Filled.Info,
|
||||
contentDescription = null,
|
||||
tint = MaterialTheme.colorScheme.onTertiaryContainer,
|
||||
modifier = Modifier.size(24.dp)
|
||||
)
|
||||
}
|
||||
Spacer(modifier = Modifier.width(12.dp))
|
||||
Column(modifier = Modifier.weight(1f)) {
|
||||
Text(
|
||||
text = "앱 버전",
|
||||
style = MaterialTheme.typography.titleSmall,
|
||||
color = MaterialTheme.colorScheme.onSurfaceVariant
|
||||
)
|
||||
Text(
|
||||
text = "v${VersionManager.getCurrentVersion(context)}",
|
||||
style = MaterialTheme.typography.titleLarge,
|
||||
fontWeight = FontWeight.Bold
|
||||
)
|
||||
}
|
||||
Button(
|
||||
onClick = {
|
||||
scope.launch {
|
||||
isCheckingUpdate = true
|
||||
val currentCode = VersionManager.getCurrentVersionCode(context)
|
||||
val remoteInfo = VersionManager.checkForUpdate()
|
||||
isCheckingUpdate = false
|
||||
|
||||
if (remoteInfo != null && VersionManager.isUpdateAvailable(currentCode, remoteInfo.versionCode)) {
|
||||
downloadAndInstallApk(context, remoteInfo)
|
||||
} else {
|
||||
Toast.makeText(context, "최신 버전입니다", Toast.LENGTH_SHORT).show()
|
||||
}
|
||||
}
|
||||
},
|
||||
enabled = !isCheckingUpdate,
|
||||
shape = RoundedCornerShape(12.dp)
|
||||
) {
|
||||
if (isCheckingUpdate) {
|
||||
CircularProgressIndicator(
|
||||
modifier = Modifier.size(18.dp),
|
||||
strokeWidth = 2.dp
|
||||
)
|
||||
} else {
|
||||
Icon(
|
||||
imageVector = Icons.Outlined.SystemUpdate,
|
||||
contentDescription = null,
|
||||
modifier = Modifier.size(18.dp)
|
||||
)
|
||||
}
|
||||
Spacer(modifier = Modifier.width(6.dp))
|
||||
Text(if (isCheckingUpdate) "확인 중..." else "업데이트 확인")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 하단 여백
|
||||
item {
|
||||
Spacer(modifier = Modifier.height(32.dp))
|
||||
}
|
||||
}
|
||||
|
||||
// 삭제 확인 다이얼로그
|
||||
if (showDeleteDialog) {
|
||||
AlertDialog(
|
||||
onDismissRequest = { showDeleteDialog = false },
|
||||
title = { Text("데이터 삭제") },
|
||||
text = { Text("저장된 모든 핫딜 데이터를 삭제하시겠습니까?\n이 작업은 되돌릴 수 없습니다.") },
|
||||
confirmButton = {
|
||||
Button(
|
||||
onClick = {
|
||||
viewModel.deleteAllParsedData()
|
||||
showDeleteDialog = false
|
||||
},
|
||||
colors = ButtonDefaults.buttonColors(
|
||||
containerColor = MaterialTheme.colorScheme.error
|
||||
)
|
||||
) {
|
||||
Text("삭제")
|
||||
}
|
||||
},
|
||||
dismissButton = {
|
||||
TextButton(onClick = { showDeleteDialog = false }) {
|
||||
Text("취소")
|
||||
}
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
private fun downloadAndInstallApk(context: Context, updateInfo: com.hotdeal.alarm.util.UpdateInfo) {
|
||||
val downloadManager = context.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager
|
||||
val request = DownloadManager.Request(Uri.parse(updateInfo.updateUrl))
|
||||
.setTitle("핫딜 알람 업데이트")
|
||||
.setDescription("버전 ${updateInfo.version} 다운로드 중...")
|
||||
.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED)
|
||||
.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, "hotdeal_alarm_${updateInfo.version}.apk")
|
||||
.setAllowedOverMetered(true)
|
||||
.setAllowedOverRoaming(true)
|
||||
|
||||
downloadManager.enqueue(request)
|
||||
Toast.makeText(context, "다운로드가 시작되었습니다", Toast.LENGTH_SHORT).show()
|
||||
}
|
||||
|
||||
// ==================== 공통 컴포넌트 ====================
|
||||
|
||||
@Composable
|
||||
private fun NotificationSettingsHeader(
|
||||
permissionStatus: PermissionHelper.PermissionStatus,
|
||||
hasEnabledSites: Boolean,
|
||||
onRequestPermission: () -> Unit,
|
||||
onOpenSystemSettings: () -> Unit
|
||||
onRequestPermission: () -> Unit
|
||||
) {
|
||||
val context = LocalContext.current
|
||||
|
||||
@@ -220,10 +508,7 @@ private fun NotificationSettingsHeader(
|
||||
),
|
||||
elevation = CardDefaults.cardElevation(defaultElevation = 2.dp)
|
||||
) {
|
||||
Column(
|
||||
modifier = Modifier.padding(16.dp)
|
||||
) {
|
||||
// 헤더
|
||||
Column(modifier = Modifier.padding(16.dp)) {
|
||||
Row(
|
||||
verticalAlignment = Alignment.CenterVertically,
|
||||
modifier = Modifier.fillMaxWidth()
|
||||
@@ -261,7 +546,6 @@ private fun NotificationSettingsHeader(
|
||||
|
||||
Spacer(modifier = Modifier.height(16.dp))
|
||||
|
||||
// 알림 권한 상태
|
||||
PermissionStatusRow(
|
||||
icon = if (permissionStatus.hasNotificationPermission) Icons.Filled.CheckCircle else Icons.Filled.Warning,
|
||||
title = "알림 권한",
|
||||
@@ -275,7 +559,6 @@ private fun NotificationSettingsHeader(
|
||||
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
|
||||
// 정확한 알람 (리마인더) 권한 상태
|
||||
PermissionStatusRow(
|
||||
icon = if (permissionStatus.hasExactAlarmPermission) Icons.Filled.CheckCircle else Icons.Filled.Warning,
|
||||
title = "리마인더 및 알람",
|
||||
@@ -289,7 +572,6 @@ private fun NotificationSettingsHeader(
|
||||
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
|
||||
// 알 수 없는 앱 설치 권한 상태
|
||||
PermissionStatusRow(
|
||||
icon = if (permissionStatus.canInstallUnknownApps) Icons.Filled.CheckCircle else Icons.Filled.Warning,
|
||||
title = "앱 설치 권한",
|
||||
@@ -303,7 +585,6 @@ private fun NotificationSettingsHeader(
|
||||
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
|
||||
// 사이트 선택 상태
|
||||
PermissionStatusRow(
|
||||
icon = if (hasEnabledSites) Icons.Filled.CheckCircle else Icons.Filled.Error,
|
||||
title = "사이트 선택",
|
||||
@@ -311,7 +592,6 @@ private fun NotificationSettingsHeader(
|
||||
isOk = hasEnabledSites
|
||||
)
|
||||
|
||||
// 시스템 설정 버튼
|
||||
if (permissionStatus.hasNotificationPermission || permissionStatus.hasExactAlarmPermission || permissionStatus.canInstallUnknownApps) {
|
||||
Spacer(modifier = Modifier.height(12.dp))
|
||||
OutlinedButton(
|
||||
@@ -371,6 +651,7 @@ private fun PermissionStatusRow(
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun SectionHeader(
|
||||
title: String,
|
||||
@@ -420,12 +701,8 @@ private fun PollingIntervalCard(
|
||||
),
|
||||
elevation = CardDefaults.cardElevation(defaultElevation = 2.dp)
|
||||
) {
|
||||
Column(
|
||||
modifier = Modifier.padding(20.dp)
|
||||
) {
|
||||
Row(
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
Column(modifier = Modifier.padding(20.dp)) {
|
||||
Row(verticalAlignment = Alignment.CenterVertically) {
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(44.dp)
|
||||
@@ -459,7 +736,6 @@ private fun PollingIntervalCard(
|
||||
|
||||
Spacer(modifier = Modifier.height(16.dp))
|
||||
|
||||
// 선택 옵션들
|
||||
val options = listOf(
|
||||
Triple(1, "1분", "빠름"),
|
||||
Triple(2, "2분", "권장"),
|
||||
@@ -524,10 +800,6 @@ private fun PollingOptionChip(
|
||||
MaterialTheme.colorScheme.primaryContainer
|
||||
else
|
||||
MaterialTheme.colorScheme.surfaceVariant.copy(alpha = 0.5f),
|
||||
border = if (isSelected)
|
||||
null
|
||||
else
|
||||
null,
|
||||
interactionSource = interactionSource
|
||||
) {
|
||||
Column(
|
||||
@@ -575,15 +847,11 @@ private fun EnhancedSiteCard(
|
||||
),
|
||||
elevation = CardDefaults.cardElevation(defaultElevation = 1.dp)
|
||||
) {
|
||||
Column(
|
||||
modifier = Modifier.padding(16.dp)
|
||||
) {
|
||||
// 사이트 헤더
|
||||
Column(modifier = Modifier.padding(16.dp)) {
|
||||
Row(
|
||||
verticalAlignment = Alignment.CenterVertically,
|
||||
modifier = Modifier.fillMaxWidth()
|
||||
) {
|
||||
// 사이트 색상 인디케이터
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(40.dp)
|
||||
@@ -612,7 +880,6 @@ private fun EnhancedSiteCard(
|
||||
)
|
||||
}
|
||||
|
||||
// 진행률 표시
|
||||
if (totalCount > 0) {
|
||||
Text(
|
||||
text = "${(enabledCount * 100 / totalCount)}%",
|
||||
@@ -623,7 +890,6 @@ private fun EnhancedSiteCard(
|
||||
}
|
||||
}
|
||||
|
||||
// 게시판 목록
|
||||
if (configs.isNotEmpty()) {
|
||||
Spacer(modifier = Modifier.height(12.dp))
|
||||
configs.forEach { config ->
|
||||
@@ -654,11 +920,8 @@ private fun EnhancedSiteCard(
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun KeywordInputCard(
|
||||
onAdd: (String) -> Unit
|
||||
) {
|
||||
private fun KeywordInputCard(onAdd: (String) -> Unit) {
|
||||
var keywordText by remember { mutableStateOf("") }
|
||||
var isExpanded by remember { mutableStateOf(false) }
|
||||
|
||||
Card(
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
@@ -667,9 +930,7 @@ private fun KeywordInputCard(
|
||||
containerColor = MaterialTheme.colorScheme.primaryContainer.copy(alpha = 0.3f)
|
||||
)
|
||||
) {
|
||||
Column(
|
||||
modifier = Modifier.padding(16.dp)
|
||||
) {
|
||||
Column(modifier = Modifier.padding(16.dp)) {
|
||||
Row(
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
@@ -713,7 +974,6 @@ private fun KeywordInputCard(
|
||||
}
|
||||
}
|
||||
|
||||
// 힌트
|
||||
AnimatedVisibility(
|
||||
visible = keywordText.isEmpty(),
|
||||
enter = fadeIn() + expandVertically(),
|
||||
@@ -743,7 +1003,7 @@ private fun EnhancedKeywordCard(
|
||||
shape = RoundedCornerShape(20.dp),
|
||||
colors = CardDefaults.elevatedCardColors(
|
||||
containerColor = if (isEnabled)
|
||||
Color(0xFFFFEBEE) // 옅은 빨간색 배경 (Material Red 50)
|
||||
Color(0xFFFFEBEE)
|
||||
else
|
||||
MaterialTheme.colorScheme.surface
|
||||
),
|
||||
@@ -757,11 +1017,10 @@ private fun EnhancedKeywordCard(
|
||||
.padding(16.dp),
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
// 키워드 아이콘
|
||||
Surface(
|
||||
shape = RoundedCornerShape(10.dp),
|
||||
color = if (isEnabled)
|
||||
Color(0xFFE53935).copy(alpha = 0.12f) // 빨간색
|
||||
Color(0xFFE53935).copy(alpha = 0.12f)
|
||||
else
|
||||
MaterialTheme.colorScheme.surfaceVariant.copy(alpha = 0.5f),
|
||||
modifier = Modifier.height(36.dp)
|
||||
@@ -790,7 +1049,6 @@ private fun EnhancedKeywordCard(
|
||||
|
||||
Spacer(modifier = Modifier.width(12.dp))
|
||||
|
||||
// 키워드 텍스트
|
||||
Column(modifier = Modifier.weight(1f)) {
|
||||
Text(
|
||||
text = keyword.keyword,
|
||||
@@ -802,17 +1060,13 @@ private fun EnhancedKeywordCard(
|
||||
text = if (isEnabled) "알림 활성화" else "알림 비활성화",
|
||||
style = MaterialTheme.typography.bodySmall,
|
||||
color = if (isEnabled)
|
||||
Color(0xFFE53935) // 빨간색
|
||||
Color(0xFFE53935)
|
||||
else
|
||||
MaterialTheme.colorScheme.onSurfaceVariant
|
||||
)
|
||||
}
|
||||
|
||||
// 액션 버튼들
|
||||
Row(
|
||||
horizontalArrangement = Arrangement.spacedBy(2.dp)
|
||||
) {
|
||||
// 토글 버튼
|
||||
Row(horizontalArrangement = Arrangement.spacedBy(2.dp)) {
|
||||
IconButton(
|
||||
onClick = onToggle,
|
||||
modifier = Modifier.size(36.dp)
|
||||
@@ -821,14 +1075,13 @@ private fun EnhancedKeywordCard(
|
||||
imageVector = if (isEnabled) Icons.Filled.Notifications else Icons.Outlined.NotificationsNone,
|
||||
contentDescription = if (isEnabled) "알림 끄기" else "알림 켜기",
|
||||
tint = if (isEnabled)
|
||||
Color(0xFFE53935) // 빨간색
|
||||
Color(0xFFE53935)
|
||||
else
|
||||
MaterialTheme.colorScheme.onSurfaceVariant,
|
||||
modifier = Modifier.size(18.dp)
|
||||
)
|
||||
}
|
||||
|
||||
// 삭제 버튼
|
||||
IconButton(
|
||||
onClick = onDelete,
|
||||
modifier = Modifier.size(36.dp)
|
||||
@@ -844,70 +1097,3 @@ private fun EnhancedKeywordCard(
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun VersionCard(
|
||||
currentVersion: String,
|
||||
onCheckUpdate: () -> Unit
|
||||
) {
|
||||
Card(
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
shape = RoundedCornerShape(16.dp),
|
||||
colors = CardDefaults.cardColors(
|
||||
containerColor = MaterialTheme.colorScheme.surface
|
||||
),
|
||||
elevation = CardDefaults.cardElevation(defaultElevation = 1.dp)
|
||||
) {
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(16.dp),
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(44.dp)
|
||||
.background(
|
||||
MaterialTheme.colorScheme.tertiaryContainer,
|
||||
CircleShape
|
||||
),
|
||||
contentAlignment = Alignment.Center
|
||||
) {
|
||||
Icon(
|
||||
imageVector = Icons.Filled.Info,
|
||||
contentDescription = null,
|
||||
tint = MaterialTheme.colorScheme.onTertiaryContainer,
|
||||
modifier = Modifier.size(24.dp)
|
||||
)
|
||||
}
|
||||
|
||||
Spacer(modifier = Modifier.width(12.dp))
|
||||
|
||||
Column(modifier = Modifier.weight(1f)) {
|
||||
Text(
|
||||
text = "앱 버전",
|
||||
style = MaterialTheme.typography.titleSmall,
|
||||
color = MaterialTheme.colorScheme.onSurfaceVariant
|
||||
)
|
||||
Text(
|
||||
text = "v$currentVersion",
|
||||
style = MaterialTheme.typography.titleLarge,
|
||||
fontWeight = FontWeight.Bold
|
||||
)
|
||||
}
|
||||
|
||||
OutlinedButton(
|
||||
onClick = onCheckUpdate,
|
||||
shape = RoundedCornerShape(12.dp)
|
||||
) {
|
||||
Icon(
|
||||
imageVector = Icons.Outlined.SystemUpdate,
|
||||
contentDescription = null,
|
||||
modifier = Modifier.size(18.dp)
|
||||
)
|
||||
Spacer(modifier = Modifier.width(6.dp))
|
||||
Text("업데이트 확인")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user