- 설정 화면에 SMS 권한 카드 추가: 상태 표시(허용/미허용/영구 거부), 권한 요청/설정 이동 지원\n- 기존 알림 권한 카드 스타일과 일관성 유지 feat(permissions): 최초 실행 시 SMS 권한 온보딩 화면 추가 및 Splash에서 라우팅 (Android) - 권한 필요 이유/수집 범위 현지화 문구 추가\n- 거부/영구거부 케이스 처리 및 설정 이동 chore(codex): AGENTS.md/체크 스크립트/CI/프롬프트 템플릿 추가 - AGENTS.md, scripts/check.sh, scripts/fix.sh, .github/workflows/flutter_ci.yml, .claude/agents/codex.md, 문서 템플릿 추가 refactor(logging): 경로별 print 제거 후 경량 로거(Log) 도입 - SMS 스캐너/컨트롤러, URL 매처, 데이터 리포지토리, 내비게이션, 메모리/성능 유틸 등 핵심 경로 치환 feat(exchange): 환율 API URL을 --dart-define로 오버라이드 가능 + 폴백 로깅 강화 test: URL 매처/환율 스모크 테스트 추가 chore(android): RECEIVE_SMS 권한 제거 (READ_SMS만 유지) fix(lints): dart fix + 수동 정리로 경고 대폭 감소, 비동기 context(mounted) 보강 fix(deprecations):\n- flutter_local_notifications의 androidAllowWhileIdle → androidScheduleMode 전환\n- WillPopScope → PopScope 교체 i18n: SMS 권한 온보딩/설정 문구 현지화 키 추가
201 lines
5.5 KiB
Dart
201 lines
5.5 KiB
Dart
import 'package:flutter/material.dart';
|
|
import '../../../theme/app_colors.dart';
|
|
|
|
/// 부차적인 액션에 사용되는 Secondary 버튼
|
|
/// 취소, 되돌아가기, 부가 옵션 등에 사용됩니다.
|
|
class SecondaryButton extends StatefulWidget {
|
|
final String text;
|
|
final VoidCallback? onPressed;
|
|
final IconData? icon;
|
|
final double? width;
|
|
final double height;
|
|
final Color? borderColor;
|
|
final Color? textColor;
|
|
final double fontSize;
|
|
final EdgeInsetsGeometry? padding;
|
|
final double borderRadius;
|
|
final double borderWidth;
|
|
final bool enableHoverEffect;
|
|
|
|
const SecondaryButton({
|
|
super.key,
|
|
required this.text,
|
|
this.onPressed,
|
|
this.icon,
|
|
this.width,
|
|
this.height = 56,
|
|
this.borderColor,
|
|
this.textColor,
|
|
this.fontSize = 16,
|
|
this.padding,
|
|
this.borderRadius = 16,
|
|
this.borderWidth = 1.5,
|
|
this.enableHoverEffect = true,
|
|
});
|
|
|
|
@override
|
|
State<SecondaryButton> createState() => _SecondaryButtonState();
|
|
}
|
|
|
|
class _SecondaryButtonState extends State<SecondaryButton> {
|
|
bool _isHovered = false;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final effectiveBorderColor = widget.borderColor ?? AppColors.secondaryColor;
|
|
final effectiveTextColor = widget.textColor ?? AppColors.primaryColor;
|
|
|
|
Widget button = AnimatedContainer(
|
|
duration: const Duration(milliseconds: 200),
|
|
width: widget.width,
|
|
height: widget.height,
|
|
transform: widget.enableHoverEffect && _isHovered
|
|
? (Matrix4.identity()..scale(1.02))
|
|
: Matrix4.identity(),
|
|
child: OutlinedButton(
|
|
onPressed: widget.onPressed,
|
|
style: OutlinedButton.styleFrom(
|
|
foregroundColor: effectiveTextColor,
|
|
shape: RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.circular(widget.borderRadius),
|
|
),
|
|
side: BorderSide(
|
|
color: _isHovered
|
|
? effectiveBorderColor.withValues(alpha: 0.4)
|
|
: effectiveBorderColor,
|
|
width: widget.borderWidth,
|
|
),
|
|
padding: widget.padding ??
|
|
const EdgeInsets.symmetric(
|
|
vertical: 12,
|
|
horizontal: 24,
|
|
),
|
|
backgroundColor:
|
|
_isHovered ? AppColors.glassBackground : Colors.transparent,
|
|
),
|
|
child: Row(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
if (widget.icon != null) ...[
|
|
Icon(
|
|
widget.icon,
|
|
color: effectiveTextColor,
|
|
size: 20,
|
|
),
|
|
const SizedBox(width: 8),
|
|
],
|
|
Text(
|
|
widget.text,
|
|
style: TextStyle(
|
|
fontSize: widget.fontSize,
|
|
fontWeight: FontWeight.w500,
|
|
color: effectiveTextColor,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
|
|
if (widget.enableHoverEffect) {
|
|
return MouseRegion(
|
|
onEnter: (_) => setState(() => _isHovered = true),
|
|
onExit: (_) => setState(() => _isHovered = false),
|
|
child: button,
|
|
);
|
|
}
|
|
|
|
return button;
|
|
}
|
|
}
|
|
|
|
/// 텍스트 링크 스타일의 버튼
|
|
/// 간단한 액션이나 링크에 사용됩니다.
|
|
class TextLinkButton extends StatefulWidget {
|
|
final String text;
|
|
final VoidCallback? onPressed;
|
|
final IconData? icon;
|
|
final Color? color;
|
|
final double fontSize;
|
|
final bool enableHoverEffect;
|
|
|
|
const TextLinkButton({
|
|
super.key,
|
|
required this.text,
|
|
this.onPressed,
|
|
this.icon,
|
|
this.color,
|
|
this.fontSize = 14,
|
|
this.enableHoverEffect = true,
|
|
});
|
|
|
|
@override
|
|
State<TextLinkButton> createState() => _TextLinkButtonState();
|
|
}
|
|
|
|
class _TextLinkButtonState extends State<TextLinkButton> {
|
|
bool _isHovered = false;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final theme = Theme.of(context);
|
|
final effectiveColor = widget.color ?? AppColors.primaryColor;
|
|
|
|
Widget button = AnimatedContainer(
|
|
duration: const Duration(milliseconds: 200),
|
|
decoration: BoxDecoration(
|
|
color: _isHovered
|
|
? theme.colorScheme.onSurface.withValues(alpha: 0.05)
|
|
: Colors.transparent,
|
|
borderRadius: BorderRadius.circular(8),
|
|
),
|
|
child: TextButton(
|
|
onPressed: widget.onPressed,
|
|
style: TextButton.styleFrom(
|
|
foregroundColor: effectiveColor,
|
|
padding: const EdgeInsets.symmetric(
|
|
horizontal: 10,
|
|
vertical: 6,
|
|
),
|
|
minimumSize: Size.zero,
|
|
tapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
|
),
|
|
child: Row(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
if (widget.icon != null) ...[
|
|
Icon(
|
|
widget.icon,
|
|
size: 18,
|
|
color: effectiveColor,
|
|
),
|
|
const SizedBox(width: 6),
|
|
],
|
|
Text(
|
|
widget.text,
|
|
style: TextStyle(
|
|
fontSize: widget.fontSize,
|
|
fontWeight: FontWeight.w500,
|
|
color: effectiveColor,
|
|
decoration:
|
|
_isHovered ? TextDecoration.underline : TextDecoration.none,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
|
|
if (widget.enableHoverEffect) {
|
|
return MouseRegion(
|
|
onEnter: (_) => setState(() => _isHovered = true),
|
|
onExit: (_) => setState(() => _isHovered = false),
|
|
child: button,
|
|
);
|
|
}
|
|
|
|
return button;
|
|
}
|
|
}
|