Files
superport_v2/lib/widgets/components/responsive.dart
JiWoong Sul d76f765814 feat(approvals): Approval Flow v2 프런트엔드 전면 개편
- 환경/라우터 모듈에 approval_flow_v2 토글을 추가하고 FeatureFlags 초기화를 연결 (.env*, lib/core/**)
- ApiClient 빌더·ApiRoutes 확장과 ApprovalRepositoryRemote 리팩터링으로 include·액션 시그니처를 정합화
- ApprovalFlow·ApprovalDraft 엔티티/레포/유즈케이스를 도입해 서버 초안과 단계 액션(승인·회수·재상신)을 지원
- Approval 컨트롤러·히스토리·템플릿 페이지와 공유 위젯을 재작성해 감사 로그·회수 UX·템플릿 CRUD를 반영
- Inbound/Outbound/Rental 컨트롤러·페이지에 결재 섹션을 삽입하고 대시보드 pending 카드 요약을 갱신
- SuperportDialog·FormField 등 공통 위젯을 보강하고 승인 위젯 가이드를 추가해 UI 가이드를 정리
- 결재/재고 테스트 픽스처와 단위·위젯·통합 테스트를 확장하고 flutter_test_config로 스테이징 호스트를 허용
- Approval Flow 레포트/플랜 문서를 업데이트하고 ApprovalFlow_System_Integration_and_ChangePlan.md를 추가
- 실행: flutter analyze, flutter test
2025-10-31 01:05:39 +09:00

136 lines
4.2 KiB
Dart

import 'package:flutter/widgets.dart';
/// 데스크톱 레이아웃으로 간주할 최소 너비(px).
const double desktopBreakpoint = 1200;
/// 태블릿 레이아웃을 구분하는 최소 너비(px).
const double tabletBreakpoint = 960;
/// 뷰포트 크기별 분기값.
enum DeviceBreakpoint { mobile, tablet, desktop }
/// 현재 화면 너비에 맞는 [DeviceBreakpoint]를 계산한다.
DeviceBreakpoint breakpointForWidth(double width) {
if (width >= desktopBreakpoint) {
return DeviceBreakpoint.desktop;
}
if (width >= tabletBreakpoint) {
return DeviceBreakpoint.tablet;
}
return DeviceBreakpoint.mobile;
}
/// 주어진 너비가 데스크톱 분기에 해당하는지 여부.
bool isDesktop(double width) => width >= desktopBreakpoint;
/// 주어진 너비가 태블릿 분기에 해당하는지 여부.
bool isTablet(double width) =>
width >= tabletBreakpoint && width < desktopBreakpoint;
/// 주어진 너비가 모바일 분기에 해당하는지 여부.
bool isMobile(double width) => width < tabletBreakpoint;
/// 컨텍스트 기반으로 데스크톱 범위인지 확인한다.
bool isDesktopContext(BuildContext context) =>
isDesktop(MediaQuery.of(context).size.width);
/// 컨텍스트 기반으로 태블릿 범위인지 확인한다.
bool isTabletContext(BuildContext context) =>
isTablet(MediaQuery.of(context).size.width);
/// 컨텍스트 기반으로 모바일 범위인지 확인한다.
bool isMobileContext(BuildContext context) =>
isMobile(MediaQuery.of(context).size.width);
/// 반응형 분기 정보를 담는 값 객체.
class ResponsiveBreakpoints {
ResponsiveBreakpoints._(this.width) : breakpoint = breakpointForWidth(width);
/// 현재 뷰 가로 너비.
final double width;
/// 너비에서 계산된 분기값.
final DeviceBreakpoint breakpoint;
/// 모바일 범위인지 여부.
bool get isMobile => breakpoint == DeviceBreakpoint.mobile;
/// 태블릿 범위인지 여부.
bool get isTablet => breakpoint == DeviceBreakpoint.tablet;
/// 데스크톱 범위인지 여부.
bool get isDesktop => breakpoint == DeviceBreakpoint.desktop;
/// 현재 컨텍스트에서 [ResponsiveBreakpoints]를 생성한다.
static ResponsiveBreakpoints of(BuildContext context) {
final size = MediaQuery.of(context).size;
return ResponsiveBreakpoints._(size.width);
}
}
/// 기기 타입에 따라 위젯 빌더를 분기 실행하는 레이아웃 헬퍼.
class ResponsiveLayoutBuilder extends StatelessWidget {
const ResponsiveLayoutBuilder({
super.key,
required this.mobile,
this.tablet,
required this.desktop,
});
/// 모바일 뷰에서 사용할 빌더.
final WidgetBuilder mobile;
/// 태블릿 뷰에서 사용할 빌더. 제공되지 않으면 데스크톱 빌더를 재사용한다.
final WidgetBuilder? tablet;
/// 데스크톱 뷰에서 사용할 빌더.
final WidgetBuilder desktop;
@override
Widget build(BuildContext context) {
return LayoutBuilder(
builder: (context, constraints) {
final breakpoint = breakpointForWidth(constraints.maxWidth);
switch (breakpoint) {
case DeviceBreakpoint.mobile:
return mobile(context);
case DeviceBreakpoint.tablet:
final tabletBuilder = tablet ?? desktop;
return tabletBuilder(context);
case DeviceBreakpoint.desktop:
return desktop(context);
}
},
);
}
}
/// 특정 분기에서만 child를 표시하는 헬퍼 위젯.
class ResponsiveVisibility extends StatelessWidget {
const ResponsiveVisibility({
super.key,
required this.child,
this.replacement = const SizedBox.shrink(),
this.visibleOn = const {
DeviceBreakpoint.mobile,
DeviceBreakpoint.tablet,
DeviceBreakpoint.desktop,
},
});
/// 조건을 만족할 때 보여줄 실제 위젯.
final Widget child;
/// 조건을 만족하지 않을 때 대체로 렌더링할 위젯.
final Widget replacement;
/// 어떤 분기에서 child를 노출할지 정의한 집합.
final Set<DeviceBreakpoint> visibleOn;
@override
Widget build(BuildContext context) {
final breakpoint = ResponsiveBreakpoints.of(context).breakpoint;
return visibleOn.contains(breakpoint) ? child : replacement;
}
}