로그인 수정, 애드몹 수정, 메인페이지까지 점검완료
@ -1,44 +1,48 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id "com.android.application"
|
id "com.android.application"
|
||||||
// START: FlutterFire Configuration
|
// (Firebase, Google Services 필요 시)
|
||||||
id 'com.google.gms.google-services'
|
id 'com.google.gms.google-services'
|
||||||
// END: FlutterFire Configuration
|
|
||||||
id "kotlin-android"
|
id "kotlin-android"
|
||||||
// The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins.
|
|
||||||
id "dev.flutter.flutter-gradle-plugin"
|
id "dev.flutter.flutter-gradle-plugin"
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
namespace = "com.allscore_app"
|
namespace = "com.allscore_app"
|
||||||
compileSdkVersion = 34
|
compileSdkVersion 34
|
||||||
ndkVersion = flutter.ndkVersion
|
|
||||||
|
|
||||||
compileOptions {
|
|
||||||
sourceCompatibility = JavaVersion.VERSION_17
|
|
||||||
targetCompatibility = JavaVersion.VERSION_17
|
|
||||||
}
|
|
||||||
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = JavaVersion.VERSION_17
|
|
||||||
}
|
|
||||||
|
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
applicationId = "com.allscore_app"
|
applicationId "com.allscore_app"
|
||||||
minSdkVersion 23
|
minSdkVersion 23
|
||||||
targetSdkVersion 34
|
targetSdkVersion 34
|
||||||
versionCode = 1
|
versionCode 1
|
||||||
versionName = "1.0"
|
versionName "1.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ...
|
||||||
|
|
||||||
buildTypes {
|
buildTypes {
|
||||||
release {
|
release {
|
||||||
// TODO: Add your own signing config for the release build.
|
signingConfig signingConfigs.debug
|
||||||
// Signing with the debug keys for now, so `flutter run --release` works.
|
|
||||||
signingConfig = signingConfigs.debug
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
compileOptions {
|
||||||
|
sourceCompatibility JavaVersion.VERSION_17
|
||||||
|
targetCompatibility JavaVersion.VERSION_17
|
||||||
|
}
|
||||||
|
kotlinOptions {
|
||||||
|
jvmTarget = JavaVersion.VERSION_17
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
flutter {
|
flutter {
|
||||||
source = "../.."
|
source = "../.."
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
// 기존 의존성 ...
|
||||||
|
implementation 'com.google.android.gms:play-services-auth:20.6.0'
|
||||||
|
}
|
||||||
|
|
||||||
|
// (Firebase Auth, Crashlytics 등을 사용한다면, 아래 구문이 필요할 수 있습니다.)
|
||||||
|
apply plugin: 'com.google.gms.google-services'
|
||||||
|
1
android/app/google-oauth-service.json
Normal file
@ -0,0 +1 @@
|
|||||||
|
{"installed":{"client_id":"19981745655-3dadv7n64jqcada6mtc1ao25k1m90gp3.apps.googleusercontent.com","project_id":"allscore-447406","auth_uri":"https://accounts.google.com/o/oauth2/auth","token_uri":"https://oauth2.googleapis.com/token","auth_provider_x509_cert_url":"https://www.googleapis.com/oauth2/v1/certs"}}
|
@ -1,27 +1,45 @@
|
|||||||
{
|
{
|
||||||
"project_info": {
|
"project_info": {
|
||||||
"project_number": "70449524223",
|
"project_number": "452355332155",
|
||||||
"firebase_url": "https://allscore-344c2-default-rtdb.asia-southeast1.firebasedatabase.app",
|
"firebase_url": "https://allscore-29edf-default-rtdb.asia-southeast1.firebasedatabase.app",
|
||||||
"project_id": "allscore-344c2",
|
"project_id": "allscore-29edf",
|
||||||
"storage_bucket": "allscore-344c2.firebasestorage.app"
|
"storage_bucket": "allscore-29edf.firebasestorage.app"
|
||||||
},
|
},
|
||||||
"client": [
|
"client": [
|
||||||
{
|
{
|
||||||
"client_info": {
|
"client_info": {
|
||||||
"mobilesdk_app_id": "1:70449524223:android:94ffb9ec98e508313e4bca",
|
"mobilesdk_app_id": "1:452355332155:android:152995468604d10d13e41e",
|
||||||
"android_client_info": {
|
"android_client_info": {
|
||||||
"package_name": "com.allscore_app"
|
"package_name": "com.allscore_app"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"oauth_client": [],
|
"oauth_client": [
|
||||||
|
{
|
||||||
|
"client_id": "452355332155-t29ceato8o62c9kq9drefe7b6hd1ka1d.apps.googleusercontent.com",
|
||||||
|
"client_type": 1,
|
||||||
|
"android_info": {
|
||||||
|
"package_name": "com.allscore_app",
|
||||||
|
"certificate_hash": "83fe36945bd0f037a7b934f9737a4fa94c47872d"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"client_id": "452355332155-jv26k1rs4tro38tc99mffid2e3gbra6j.apps.googleusercontent.com",
|
||||||
|
"client_type": 3
|
||||||
|
}
|
||||||
|
],
|
||||||
"api_key": [
|
"api_key": [
|
||||||
{
|
{
|
||||||
"current_key": "AIzaSyAJEItMxO-TemHGlveSKySG-eNaTD9XJI0"
|
"current_key": "AIzaSyB6hil7Nrk8wslHDfRNRRyw6rQktY16tTc"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"services": {
|
"services": {
|
||||||
"appinvite_service": {
|
"appinvite_service": {
|
||||||
"other_platform_oauth_client": []
|
"other_platform_oauth_client": [
|
||||||
|
{
|
||||||
|
"client_id": "452355332155-jv26k1rs4tro38tc99mffid2e3gbra6j.apps.googleusercontent.com",
|
||||||
|
"client_type": 3
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,16 +2,24 @@
|
|||||||
package="com.allscore_app">
|
package="com.allscore_app">
|
||||||
|
|
||||||
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
|
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
|
||||||
|
<uses-permission android:name="android.permission.INTERNET" />
|
||||||
|
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:label="allscore_app"
|
android:label="올스코어"
|
||||||
android:name="${applicationName}"
|
android:name="${applicationName}"
|
||||||
android:icon="@mipmap/ic_launcher">
|
android:icon="@mipmap/ic_launcher">
|
||||||
|
|
||||||
<!-- ★ 여기에 meta-data 추가 ★ -->
|
<!-- ★ 여기에 meta-data 추가 ★ -->
|
||||||
<meta-data
|
<meta-data
|
||||||
android:name="com.google.android.gms.ads.APPLICATION_ID"
|
android:name="com.google.android.gms.ads.APPLICATION_ID"
|
||||||
android:value="ca-app-pub-3151339278746301~2596989191" />
|
android:value="ca-app-pub-3940256099942544~3347511713" />
|
||||||
|
<!-- android:value="ca-app-pub-6461991944599918~9492697896" -->
|
||||||
|
|
||||||
|
<!-- 구글 로그인 관련 -->
|
||||||
|
<meta-data
|
||||||
|
android:name="com.google.android.gms.auth.api.signin.client_id"
|
||||||
|
android:value="19981745655-3dadv7n64jqcada6mtc1ao25k1m90gp3.apps.googleusercontent.com" />
|
||||||
|
|
||||||
<activity
|
<activity
|
||||||
android:name=".MainActivity"
|
android:name=".MainActivity"
|
||||||
|
@ -10,6 +10,7 @@ buildscript {
|
|||||||
classpath "com.android.tools.build:gradle:8.2.1"
|
classpath "com.android.tools.build:gradle:8.2.1"
|
||||||
// 만약 Kotlin 버전 등의 추가 classpath가 필요하면 여기에 추가
|
// 만약 Kotlin 버전 등의 추가 classpath가 필요하면 여기에 추가
|
||||||
// classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:1.8.10"
|
// classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:1.8.10"
|
||||||
|
classpath 'com.google.gms:google-services:4.4.2'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BIN
assets/images/icons8-google-logo-144.png
Normal file
After Width: | Height: | Size: 3.9 KiB |
BIN
assets/images/icons8-google-logo-192.png
Normal file
After Width: | Height: | Size: 5.2 KiB |
BIN
assets/images/icons8-google-logo-36.png
Normal file
After Width: | Height: | Size: 972 B |
BIN
assets/images/icons8-google-logo-48-2.png
Normal file
After Width: | Height: | Size: 1.3 KiB |
BIN
assets/images/icons8-google-logo-48.png
Normal file
After Width: | Height: | Size: 1.3 KiB |
BIN
assets/images/icons8-google-logo-72.png
Normal file
After Width: | Height: | Size: 1.9 KiB |
BIN
assets/images/icons8-google-logo-96.png
Normal file
After Width: | Height: | Size: 2.6 KiB |
@ -5,7 +5,7 @@
|
|||||||
<key>CFBundleDevelopmentRegion</key>
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||||
<key>CFBundleDisplayName</key>
|
<key>CFBundleDisplayName</key>
|
||||||
<string>Allscore App</string>
|
<string>올스코어</string>
|
||||||
<key>CFBundleExecutable</key>
|
<key>CFBundleExecutable</key>
|
||||||
<string>$(EXECUTABLE_NAME)</string>
|
<string>$(EXECUTABLE_NAME)</string>
|
||||||
<key>CFBundleIdentifier</key>
|
<key>CFBundleIdentifier</key>
|
||||||
|
4
lib/config/config.dart
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
class Config {
|
||||||
|
static const String testAdUnitId = 'ca-app-pub-3940256099942544/6300978111';
|
||||||
|
static const String realAdUnitId = 'ca-app-pub-3940256099942544/6300978111';
|
||||||
|
}
|
149
lib/dialogs/room_setting_finish_dialog.dart
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
/// 종료된 방에서 "방 정보 보기" 버튼을 누르면 표시되는 읽기전용 모달 (수정판)
|
||||||
|
class RoomSettingFinishDialog extends StatelessWidget {
|
||||||
|
final Map<String, dynamic> roomInfo;
|
||||||
|
|
||||||
|
const RoomSettingFinishDialog({
|
||||||
|
Key? key,
|
||||||
|
required this.roomInfo,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final media = MediaQuery.of(context);
|
||||||
|
final String roomTitle = (roomInfo['room_title'] ?? '') as String;
|
||||||
|
final String roomIntro = (roomInfo['room_intro'] ?? '') as String;
|
||||||
|
final String openYn = (roomInfo['open_yn'] ?? 'Y') as String;
|
||||||
|
final int maxPeople = (roomInfo['number_of_people'] ?? 0) as int;
|
||||||
|
final int runningTime = (roomInfo['running_time'] ?? 0) as int;
|
||||||
|
final String scoreOpen = (roomInfo['score_open_range'] ?? 'ALL') as String;
|
||||||
|
|
||||||
|
final String openLabel = (openYn == 'Y') ? '공개' : '비공개';
|
||||||
|
|
||||||
|
return Dialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(18)),
|
||||||
|
child: ConstrainedBox(
|
||||||
|
constraints: BoxConstraints(
|
||||||
|
maxHeight: media.size.height * 0.8,
|
||||||
|
minWidth: media.size.width * 0.8,
|
||||||
|
),
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.all(20),
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
// (A) 상단 타이틀
|
||||||
|
const Text(
|
||||||
|
'방 정보',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 18,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Colors.black,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Divider(color: Colors.black54),
|
||||||
|
const SizedBox(height: 12),
|
||||||
|
|
||||||
|
// (B) 방 제목
|
||||||
|
_buildLabelValue(label: '방 제목', value: roomTitle.isNotEmpty ? roomTitle : '(없음)'),
|
||||||
|
const SizedBox(height: 14),
|
||||||
|
|
||||||
|
// (C) 방 소개
|
||||||
|
const Text(
|
||||||
|
'방 소개',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 15,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Colors.black87,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 6),
|
||||||
|
|
||||||
|
// 가로로 꽉 차게(width: double.infinity), 세로 최대 100px → 스크롤
|
||||||
|
Container(
|
||||||
|
width: double.infinity,
|
||||||
|
constraints: const BoxConstraints(maxHeight: 100),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
border: Border.all(color: Colors.black26),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
),
|
||||||
|
padding: const EdgeInsets.all(8),
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
child: Text(
|
||||||
|
roomIntro.isNotEmpty ? roomIntro : '소개글 없음',
|
||||||
|
style: const TextStyle(fontSize: 14, color: Colors.black),
|
||||||
|
softWrap: true,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
|
||||||
|
// (D) 공개 여부
|
||||||
|
_buildLabelValue(label: '공개 여부', value: openLabel),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
|
||||||
|
// (E) 최대 인원
|
||||||
|
_buildLabelValue(label: '최대 인원', value: '$maxPeople 명'),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
|
||||||
|
// (F) 운영 시간
|
||||||
|
_buildLabelValue(label: '운영 시간', value: '$runningTime 분'),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
|
||||||
|
// (G) 점수 공개 범위
|
||||||
|
_buildLabelValue(label: '점수 공개 범위', value: scoreOpen),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
|
||||||
|
// (H) 하단 닫기 버튼 (가운데 정렬)
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: () => Navigator.pop(context),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
),
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
|
||||||
|
),
|
||||||
|
child: const Text('닫기'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// label + value를 세로로 배치한 간단 위젯
|
||||||
|
Widget _buildLabelValue({
|
||||||
|
required String label,
|
||||||
|
required String value,
|
||||||
|
}) {
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
label,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 15,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Colors.black87,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Text(
|
||||||
|
value,
|
||||||
|
style: const TextStyle(fontSize: 14, color: Colors.black),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -69,8 +69,11 @@ void showSettingsDialog(BuildContext context) {
|
|||||||
// 로그아웃 클릭 시 동작
|
// 로그아웃 클릭 시 동작
|
||||||
SharedPreferences prefs = await SharedPreferences.getInstance();
|
SharedPreferences prefs = await SharedPreferences.getInstance();
|
||||||
await prefs.setString('auth_token', ''); // auth_token 초기화
|
await prefs.setString('auth_token', ''); // auth_token 초기화
|
||||||
Navigator.of(context).pushReplacement(
|
await prefs.setBool('auto_login', false); // auto_login 초기화
|
||||||
|
Navigator.pushAndRemoveUntil(
|
||||||
|
context,
|
||||||
MaterialPageRoute(builder: (context) => const LoginPage()), // 로그인 페이지로 이동
|
MaterialPageRoute(builder: (context) => const LoginPage()), // 로그인 페이지로 이동
|
||||||
|
(route) => false,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
style: ButtonStyle(
|
style: ButtonStyle(
|
||||||
|
127
lib/dialogs/user_info_finish_dialog.dart
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
class UserInfoFinishDialog extends StatelessWidget {
|
||||||
|
final Map<String, dynamic> userData;
|
||||||
|
|
||||||
|
const UserInfoFinishDialog({Key? key, required this.userData}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
// 예: { "nickname": "testuser4", "profile_img": "...", "introduce_myself": "..." }
|
||||||
|
final nickname = (userData['nickname'] ?? '').toString();
|
||||||
|
final profileImg = (userData['profile_img'] ?? '').toString();
|
||||||
|
final intro = (userData['introduce_myself'] ?? '').toString().trim();
|
||||||
|
|
||||||
|
return Dialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(16)),
|
||||||
|
child: Padding(
|
||||||
|
// 좌우여백 좀 더 주고, 세로도 여유롭게
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 24),
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
// (A) 상단 타이틀
|
||||||
|
const Text(
|
||||||
|
'유저 정보',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 18,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
|
||||||
|
// (B) 프로필 이미지
|
||||||
|
Container(
|
||||||
|
width: 100,
|
||||||
|
height: 100,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border: Border.all(color: Colors.black26, width: 1.5),
|
||||||
|
),
|
||||||
|
child: ClipOval(
|
||||||
|
child: (profileImg.isNotEmpty)
|
||||||
|
? Image.network(
|
||||||
|
'https://eldsoft.com:8097/images$profileImg',
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
errorBuilder: (ctx, err, st) =>
|
||||||
|
const Center(child: Text('이미지\n오류', textAlign: TextAlign.center, style: TextStyle(fontSize: 12))),
|
||||||
|
)
|
||||||
|
: const Center(
|
||||||
|
child: Text(
|
||||||
|
'이미지\n없음',
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: TextStyle(fontSize: 12),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
|
||||||
|
// (C) 닉네임
|
||||||
|
Text(
|
||||||
|
nickname.isNotEmpty ? nickname : '유저',
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
|
||||||
|
// (D) 소개 라벨
|
||||||
|
const Align(
|
||||||
|
alignment: Alignment.centerLeft,
|
||||||
|
child: Text(
|
||||||
|
'소개',
|
||||||
|
style: TextStyle(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
fontSize: 14,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 6),
|
||||||
|
|
||||||
|
// (E) 소개 내용 (길이에 맞춰 자연스럽게 높이 확장)
|
||||||
|
Container(
|
||||||
|
width: double.infinity,
|
||||||
|
// 최소 60, 최대 200 정도로 제한
|
||||||
|
constraints: const BoxConstraints(
|
||||||
|
minHeight: 60,
|
||||||
|
maxHeight: 200,
|
||||||
|
),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
border: Border.all(color: Colors.black26),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
),
|
||||||
|
padding: const EdgeInsets.all(8),
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
child: Text(
|
||||||
|
intro.isNotEmpty ? intro : '소개글이 없습니다.',
|
||||||
|
style: const TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
|
||||||
|
// (F) 확인 버튼
|
||||||
|
SizedBox(
|
||||||
|
width: 100,
|
||||||
|
child: ElevatedButton(
|
||||||
|
onPressed: () => Navigator.pop(context),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 12),
|
||||||
|
),
|
||||||
|
child: const Text('확인', style: TextStyle(fontSize: 14)),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -11,13 +11,17 @@ import 'firebase_options.dart';
|
|||||||
import 'views/login/login_page.dart';
|
import 'views/login/login_page.dart';
|
||||||
import 'views/room/main_page.dart';
|
import 'views/room/main_page.dart';
|
||||||
|
|
||||||
|
// 모바일 광고
|
||||||
|
import 'package:google_mobile_ads/google_mobile_ads.dart';
|
||||||
|
|
||||||
void main() async {
|
void main() async {
|
||||||
WidgetsFlutterBinding.ensureInitialized();
|
WidgetsFlutterBinding.ensureInitialized();
|
||||||
|
|
||||||
// Firebase 초기화
|
// 파이어베이스 초기화
|
||||||
await Firebase.initializeApp(
|
await Firebase.initializeApp();
|
||||||
options: DefaultFirebaseOptions.currentPlatform, // FirebaseOptions 사용
|
|
||||||
);
|
// 모바일 광고 초기화
|
||||||
|
MobileAds.instance.initialize();
|
||||||
|
|
||||||
runApp(const MyApp());
|
runApp(const MyApp());
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,12 @@ import 'login_page.dart';
|
|||||||
import 'pw_finding_page.dart';
|
import 'pw_finding_page.dart';
|
||||||
import 'signup_page.dart';
|
import 'signup_page.dart';
|
||||||
|
|
||||||
|
// 모바일 광고
|
||||||
|
import 'package:google_mobile_ads/google_mobile_ads.dart';
|
||||||
|
|
||||||
|
// 설정
|
||||||
|
import '../../config/config.dart';
|
||||||
|
|
||||||
class IdFindingPage extends StatefulWidget {
|
class IdFindingPage extends StatefulWidget {
|
||||||
const IdFindingPage({Key? key}) : super(key: key);
|
const IdFindingPage({Key? key}) : super(key: key);
|
||||||
|
|
||||||
@ -21,7 +27,14 @@ class _IdFindingPageState extends State<IdFindingPage> {
|
|||||||
String foundIdMessage = '';
|
String foundIdMessage = '';
|
||||||
String authId = '';
|
String authId = '';
|
||||||
|
|
||||||
|
/// (1) 광고 배너 관련 변수
|
||||||
|
BannerAd? _bannerAd;
|
||||||
|
bool _isBannerReady = false; // 광고 로드 완료 여부
|
||||||
|
String adUnitId = Config.testAdUnitId;
|
||||||
|
|
||||||
Future<void> _findId(String nickname, String email) async {
|
Future<void> _findId(String nickname, String email) async {
|
||||||
|
|
||||||
|
|
||||||
// 로딩 인디케이터 표시
|
// 로딩 인디케이터 표시
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
@ -85,6 +98,36 @@ class _IdFindingPageState extends State<IdFindingPage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_initBannerAd();
|
||||||
|
}
|
||||||
|
|
||||||
|
void _initBannerAd() {
|
||||||
|
_bannerAd = BannerAd(
|
||||||
|
// 실제/테스트 배너 광고 단위 ID
|
||||||
|
adUnitId: adUnitId,
|
||||||
|
size: AdSize.banner,
|
||||||
|
request: const AdRequest(),
|
||||||
|
listener: BannerAdListener(
|
||||||
|
onAdLoaded: (ad) {
|
||||||
|
setState(() => _isBannerReady = true);
|
||||||
|
},
|
||||||
|
onAdFailedToLoad: (ad, error) {
|
||||||
|
ad.dispose();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
_bannerAd?.load();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_bannerAd?.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> _findAllId() async {
|
Future<void> _findAllId() async {
|
||||||
// ID 전체 찾기 요청 처리
|
// ID 전체 찾기 요청 처리
|
||||||
print('ID 전체 찾기 요청 $authId'); // 요청 시 출력
|
print('ID 전체 찾기 요청 $authId'); // 요청 시 출력
|
||||||
@ -185,7 +228,6 @@ class _IdFindingPageState extends State<IdFindingPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
@ -303,10 +345,21 @@ class _IdFindingPageState extends State<IdFindingPage> {
|
|||||||
},
|
},
|
||||||
child: const Text('회원가입', style: TextStyle(color: Colors.black)),
|
child: const Text('회원가입', style: TextStyle(color: Colors.black)),
|
||||||
),
|
),
|
||||||
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// (3) 하단 광고 영역
|
||||||
|
bottomNavigationBar: _isBannerReady && _bannerAd != null
|
||||||
|
? Container(
|
||||||
|
color: Colors.white,
|
||||||
|
width: _bannerAd!.size.width.toDouble(),
|
||||||
|
height: _bannerAd!.size.height.toDouble(),
|
||||||
|
child: AdWidget(ad: _bannerAd!),
|
||||||
|
)
|
||||||
|
: SizedBox.shrink(), // 로딩 전엔 빈 공간 or 원하는 위젯
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,50 +1,118 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:http/http.dart' as http;
|
// import 'package:http/http.dart' as http; ← 사용안함. Api.serverRequest() 사용
|
||||||
import 'dart:convert';
|
import 'dart:convert' show utf8, jsonEncode;
|
||||||
import 'dart:convert' show utf8;
|
|
||||||
import 'package:crypto/crypto.dart';
|
import 'package:crypto/crypto.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
|
// 우리의 Api 모듈
|
||||||
|
import '../../plugins/api.dart';
|
||||||
|
|
||||||
|
// 안내 모달창
|
||||||
|
import '../../dialogs/response_dialog.dart';
|
||||||
|
|
||||||
|
// (기존) ID/PW 찾기, 회원가입 페이지
|
||||||
import 'id_finding_page.dart';
|
import 'id_finding_page.dart';
|
||||||
import 'pw_finding_page.dart';
|
import 'pw_finding_page.dart';
|
||||||
import 'signup_page.dart';
|
import 'signup_page.dart';
|
||||||
|
|
||||||
|
// 구글 로그인
|
||||||
|
import 'package:google_sign_in/google_sign_in.dart';
|
||||||
|
import 'package:firebase_auth/firebase_auth.dart';
|
||||||
|
|
||||||
|
// 광고
|
||||||
import 'package:google_mobile_ads/google_mobile_ads.dart';
|
import 'package:google_mobile_ads/google_mobile_ads.dart';
|
||||||
|
|
||||||
|
// 메인 페이지
|
||||||
import '../room/main_page.dart';
|
import '../room/main_page.dart';
|
||||||
|
|
||||||
|
// 설정
|
||||||
|
import '../../config/config.dart';
|
||||||
|
|
||||||
|
// 뒤로가기
|
||||||
|
import 'package:fluttertoast/fluttertoast.dart'; // 뒤로가기 안내 문구에 Toast 등 사용
|
||||||
|
|
||||||
class LoginPage extends StatefulWidget {
|
class LoginPage extends StatefulWidget {
|
||||||
const LoginPage({Key? key}) : super(key: key);
|
const LoginPage({Key? key}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
_LoginPageState createState() => _LoginPageState();
|
State<LoginPage> createState() => _LoginPageState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _LoginPageState extends State<LoginPage> {
|
class _LoginPageState extends State<LoginPage> {
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (A) ID/PW 관련
|
||||||
|
// ─────────────────────────────────────────
|
||||||
final TextEditingController idController = TextEditingController();
|
final TextEditingController idController = TextEditingController();
|
||||||
final TextEditingController passwordController = TextEditingController();
|
final TextEditingController passwordController = TextEditingController();
|
||||||
bool autoLogin = false;
|
bool autoLogin = false;
|
||||||
String loginErrorMessage = '';
|
String loginErrorMessage = ''; // 로그인 실패 시 안내
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (B) 구글 로그인 객체
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
final GoogleSignIn _googleSignIn = GoogleSignIn(
|
||||||
|
scopes: <String>['email'],
|
||||||
|
);
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (C) 광고 배너
|
||||||
|
// ─────────────────────────────────────────
|
||||||
BannerAd? _bannerAd;
|
BannerAd? _bannerAd;
|
||||||
Widget? _adWidget;
|
bool _isBannerReady = false;
|
||||||
|
String adUnitId = Config.testAdUnitId;
|
||||||
|
|
||||||
|
// 뒤로가기 처리
|
||||||
|
DateTime? _lastPressedTime;
|
||||||
|
|
||||||
|
// 로딩 중
|
||||||
|
bool _isLoading = false;
|
||||||
|
|
||||||
|
// 예: 2초 이내로 뒤로가기를 한 번 더 누르면 종료
|
||||||
|
static const _exitDuration = Duration(seconds: 2);
|
||||||
|
|
||||||
|
Future<bool> _onWillPop() async {
|
||||||
|
final now = DateTime.now();
|
||||||
|
if (_lastPressedTime == null ||
|
||||||
|
now.difference(_lastPressedTime!) > _exitDuration) {
|
||||||
|
// 첫 번째 뒤로가기 누름 or 이전 누름이 오래 전
|
||||||
|
_lastPressedTime = now;
|
||||||
|
|
||||||
|
// 안내 문구 띄우기 (Toast 예시)
|
||||||
|
Fluttertoast.showToast(
|
||||||
|
msg: '한 번 더 누르면 앱이 종료됩니다.',
|
||||||
|
toastLength: Toast.LENGTH_SHORT,
|
||||||
|
gravity: ToastGravity.BOTTOM,
|
||||||
|
);
|
||||||
|
return false; // 페이지 pop 하지 않음
|
||||||
|
}
|
||||||
|
// 2초 이내에 뒤로가기 두 번째 누름 → 앱 종료 허용
|
||||||
|
return true; // pop 허용 (Scaffold 밖으로 벗어남, 결과적으로 앱 종료)
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
_initBannerAd();
|
||||||
|
}
|
||||||
|
|
||||||
|
void _initBannerAd() {
|
||||||
_bannerAd = BannerAd(
|
_bannerAd = BannerAd(
|
||||||
adUnitId: "ca-app-pub-3151339278746301~1689299887",
|
// 실제/테스트 배너 광고 단위 ID
|
||||||
request: const AdRequest(),
|
adUnitId: adUnitId,
|
||||||
size: AdSize.banner,
|
size: AdSize.banner,
|
||||||
|
request: const AdRequest(),
|
||||||
listener: BannerAdListener(
|
listener: BannerAdListener(
|
||||||
onAdLoaded: (ad) {
|
onAdLoaded: (ad) {
|
||||||
setState(() {
|
setState(() => _isBannerReady = true);
|
||||||
_adWidget = AdWidget(ad: ad as AdWithView);
|
print('로그인페이지 배너 광고 로드 완료');
|
||||||
});
|
|
||||||
},
|
},
|
||||||
onAdFailedToLoad: (ad, error) {
|
onAdFailedToLoad: (ad, error) {
|
||||||
print('Ad failed to load: $error');
|
print('로그인페이지 배너 광고 로드 실패: $error');
|
||||||
ad.dispose();
|
ad.dispose();
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
)..load();
|
);
|
||||||
|
_bannerAd?.load();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -53,93 +121,287 @@ class _LoginPageState extends State<LoginPage> {
|
|||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _login() async {
|
// ─────────────────────────────────────────
|
||||||
String id = idController.text.trim();
|
// (D1) ID/PW 로그인
|
||||||
String password = passwordController.text.trim();
|
// ─────────────────────────────────────────
|
||||||
|
Future<void> _loginWithIdPw() async {
|
||||||
|
setState(() => _isLoading = true);
|
||||||
|
final id = idController.text.trim();
|
||||||
|
final pw = passwordController.text.trim();
|
||||||
|
|
||||||
// autoLogin 체크여부
|
// PW SHA-256 해싱
|
||||||
String autoLoginStatus = autoLogin ? 'Y' : 'N';
|
final bytes = utf8.encode(pw);
|
||||||
|
final digest = sha256.convert(bytes);
|
||||||
|
final hashedPw = digest.toString();
|
||||||
|
|
||||||
// PW를 sha256으로 해시
|
final requestBody = {
|
||||||
var bytes = utf8.encode(password);
|
"user_id": id,
|
||||||
var digest = sha256.convert(bytes);
|
"user_pw": hashedPw,
|
||||||
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final response = await http.post(
|
// (1) /user/login 서버 요청
|
||||||
Uri.parse('https://eldsoft.com:8097/user/login'),
|
final response = await Api.serverRequest(uri: '/user/login', body: requestBody);
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
'auth_token': '',
|
|
||||||
},
|
|
||||||
body: jsonEncode({
|
|
||||||
'user_id': id,
|
|
||||||
'user_pw': digest.toString(),
|
|
||||||
}),
|
|
||||||
).timeout(const Duration(seconds: 10));
|
|
||||||
|
|
||||||
// 응답 바디 디코딩
|
if (response['result'] == 'OK') {
|
||||||
String responseBody = utf8.decode(response.bodyBytes);
|
// 내부 응답
|
||||||
|
final resp = response['response'] ?? {};
|
||||||
if (response.statusCode == 200) {
|
if (resp['result'] == 'OK') {
|
||||||
final Map<String, dynamic> jsonResponse = jsonDecode(responseBody);
|
|
||||||
print('jsonResponse: $jsonResponse');
|
|
||||||
|
|
||||||
if (jsonResponse['result'] == 'OK') {
|
|
||||||
// 로그인 성공
|
// 로그인 성공
|
||||||
final authData = jsonResponse['auth'] ?? {};
|
print('ID/PW 로그인 성공: $resp');
|
||||||
final token = authData['token'] ?? '';
|
|
||||||
final userSeq = authData['user_seq'] ?? 0; // 새로 추가
|
|
||||||
|
|
||||||
SharedPreferences prefs = await SharedPreferences.getInstance();
|
// (a) google_user_yn = N
|
||||||
// 토큰 및 autoLogin 여부 저장
|
final prefs = await SharedPreferences.getInstance();
|
||||||
await prefs.setString('auth_token', token);
|
await prefs.setString('oauth_type', 'idpw');
|
||||||
await prefs.setBool('auto_login', autoLogin);
|
await prefs.setBool('auto_login', true);
|
||||||
// (New) 내 user_seq 저장
|
await prefs.setString('jwt_token', resp['auth']['token'].toString());
|
||||||
await prefs.setInt('my_user_seq', userSeq);
|
await prefs.setString('user_seq', resp['auth']['user_seq'].toString());
|
||||||
|
|
||||||
// 메인 페이지로 이동
|
// 메인 페이지 이동
|
||||||
|
if (!mounted) return;
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(builder: (context) => const MainPage()),
|
MaterialPageRoute(builder: (_) => const MainPage()),
|
||||||
);
|
);
|
||||||
} else if (jsonResponse['response_info']['msg_title'] == '로그인 실패') {
|
|
||||||
// 로그인 실패 메시지
|
|
||||||
setState(() {
|
|
||||||
loginErrorMessage = '회원정보를 다시 확인해주세요.';
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
// result != OK 이지만, 다른 이유
|
showResponseDialog(context, resp['response_info']['msg_title'], resp['response_info']['msg_content']);
|
||||||
_showDialog('로그인 실패', '서버에서 로그인에 실패했습니다.\n관리자에게 문의해주세요.');
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
_showDialog('오류', '로그인에 실패했습니다. 관리자에게 문의해주세요.');
|
// 서버 통신 자체가 FAIL
|
||||||
|
showResponseDialog(context, '오류', '로그인 요청 실패');
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print('로그인 요청 중 오류: $e');
|
showResponseDialog(context, '오류', '로그인 요청 중 예외 발생.\n$e');
|
||||||
_showDialog('오류', '로그인 요청이 실패했습니다. 관리자에게 문의해주세요.\n$e');
|
} finally {
|
||||||
|
setState(() => _isLoading = false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showDialog(String title, String content) {
|
// ─────────────────────────────────────────
|
||||||
showDialog(
|
// (D2) 구글 로그인
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
Future<void> _googleLogin() async {
|
||||||
|
setState(() => _isLoading = true);
|
||||||
|
try {
|
||||||
|
// 1) 구글 계정 선택
|
||||||
|
final GoogleSignInAccount? googleUser = await _googleSignIn.signIn();
|
||||||
|
if (googleUser == null) {
|
||||||
|
// 사용자가 로그인 창에서 취소 누름
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2) 구글 인증 정보 가져오기
|
||||||
|
final GoogleSignInAuthentication googleAuth = await googleUser.authentication;
|
||||||
|
|
||||||
|
// 3) FirebaseAuth Credential 생성
|
||||||
|
final AuthCredential credential = GoogleAuthProvider.credential(
|
||||||
|
accessToken: googleAuth.accessToken,
|
||||||
|
idToken: googleAuth.idToken,
|
||||||
|
);
|
||||||
|
|
||||||
|
// 4) FirebaseAuth로 로그인
|
||||||
|
final UserCredential userCredential =
|
||||||
|
await FirebaseAuth.instance.signInWithCredential(credential);
|
||||||
|
|
||||||
|
final User? user = userCredential.user;
|
||||||
|
if (user == null) {
|
||||||
|
print('구글 로그인 실패: Firebase User가 null');
|
||||||
|
_showAlert('로그인 오류', 'Firebase User가 null입니다.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final idToken = await user.getIdToken();
|
||||||
|
|
||||||
|
// 서버에 구글 로그인 정보 전송
|
||||||
|
final requestBody = {
|
||||||
|
'id_token': idToken,
|
||||||
|
};
|
||||||
|
|
||||||
|
final response = await Api.serverRequest(uri: '/user/google/login', body: requestBody);
|
||||||
|
if (response['result'] == 'OK') {
|
||||||
|
final resp = response['response'] ?? {};
|
||||||
|
if (resp['result'] == 'OK') {
|
||||||
|
final prefs = await SharedPreferences.getInstance();
|
||||||
|
await prefs.setString('oauth_type', 'google');
|
||||||
|
await prefs.setBool('auto_login', true);
|
||||||
|
await prefs.setString('jwt_token', resp['auth']['token'].toString());
|
||||||
|
await prefs.setString('user_seq', resp['auth']['user_seq'].toString());
|
||||||
|
Navigator.pushReplacement(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(builder: (_) => const MainPage()),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
showResponseDialog(context, resp['response_info']['msg_title'], resp['response_info']['msg_content']);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showResponseDialog(context, '오류', '구글 로그인 요청 실패');
|
||||||
|
}
|
||||||
|
|
||||||
|
// (선택) SharedPreferences에 google_user_yn = 'Y' 저장 등
|
||||||
|
final prefs = await SharedPreferences.getInstance();
|
||||||
|
await prefs.setString('google_user_yn', 'Y');
|
||||||
|
} catch (e) {
|
||||||
|
_showAlert('오류', '구글 로그인 중 오류가 발생했습니다.\n$e');
|
||||||
|
} finally {
|
||||||
|
setState(() => _isLoading = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (D3) 구글 회원가입
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
Future<void> _googleSignUp() async {
|
||||||
|
final agreed = await _showTermsModal();
|
||||||
|
if (agreed != true) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// (2) 구글 계정 선택
|
||||||
|
final GoogleSignInAccount? googleUser = await _googleSignIn.signIn();
|
||||||
|
if (googleUser == null) {
|
||||||
|
// 사용자가 회원가입 창에서 취소 누름
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// (3) 구글 인증 정보 가져오기
|
||||||
|
final GoogleSignInAuthentication googleAuth = await googleUser.authentication;
|
||||||
|
|
||||||
|
// (4) FirebaseAuth Credential 생성
|
||||||
|
final AuthCredential credential = GoogleAuthProvider.credential(
|
||||||
|
accessToken: googleAuth.accessToken,
|
||||||
|
idToken: googleAuth.idToken,
|
||||||
|
);
|
||||||
|
|
||||||
|
// (5) FirebaseAuth로 로그인 (회원가입 시도)
|
||||||
|
// 사실 "회원가입"이라는 명칭을 썼지만, Firebase 쪽에선 같은 signInWithCredential()를 사용
|
||||||
|
final UserCredential userCredential =
|
||||||
|
await FirebaseAuth.instance.signInWithCredential(credential);
|
||||||
|
|
||||||
|
final User? user = userCredential.user;
|
||||||
|
if (user == null) {
|
||||||
|
showResponseDialog(context, '오류', '구글계정 인증에 실패했습니다.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// (6) idToken 추출 후, 서버에 회원가입 요청
|
||||||
|
final idToken = await user.getIdToken();
|
||||||
|
final requestBody = {
|
||||||
|
'id_token': idToken,
|
||||||
|
};
|
||||||
|
|
||||||
|
// 서버 측 '/user/google/signup' API 호출 (예시)
|
||||||
|
final response = await Api.serverRequest(uri: '/user/google/signup', body: requestBody);
|
||||||
|
if (response['result'] == 'OK') {
|
||||||
|
final resp = response['response'] ?? {};
|
||||||
|
if (resp['result'] == 'OK') {
|
||||||
|
// 회원가입 성공 안내
|
||||||
|
showResponseDialog(context, '회원가입 완료', '구글 회원가입이 완료되었습니다.');
|
||||||
|
} else {
|
||||||
|
// 서버 응답 자체가 OK가 아니면, 어떤 에러 메시지를 띄울지 처리
|
||||||
|
final msgTitle = resp['response_info']?['msg_title'] ?? '오류';
|
||||||
|
final msgContent = resp['response_info']?['msg_content'] ?? '회원가입에 실패했습니다.';
|
||||||
|
showResponseDialog(context, msgTitle, msgContent);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 서버 통신이 FAIL (응답 result != OK)
|
||||||
|
showResponseDialog(context, '오류', '구글 회원가입 요청 실패');
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
showResponseDialog(context, '오류', '구글 회원가입 중 오류가 발생했습니다.\n$e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (E) 약관 모달
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
Future<bool?> _showTermsModal() async {
|
||||||
|
return showDialog<bool>(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext context) {
|
barrierDismissible: false,
|
||||||
|
builder: (ctx) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
title: Text(title, style: const TextStyle(color: Colors.black)),
|
title: const Text(
|
||||||
content: Text(content, style: const TextStyle(color: Colors.black)),
|
'개인정보 수집 및 이용 동의서',
|
||||||
actions: <Widget>[
|
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
||||||
Center(
|
),
|
||||||
child: TextButton(
|
content: SingleChildScrollView(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: const [
|
||||||
|
Text(
|
||||||
|
'''올스코어(이하 "회사"라 합니다)는 이용자의 개인정보를 중요시하며, 「개인정보 보호법」 등 관련 법령을 준수하고 있습니다. 회사는 개인정보 수집 및 이용에 관한 사항을 아래와 같이 안내드리오니, 내용을 충분히 숙지하신 후 동의하여 주시기 바랍니다.
|
||||||
|
|
||||||
|
1. 수집하는 개인정보 항목
|
||||||
|
필수 항목: 아이디(ID), 비밀번호(PW), 닉네임(실명 아님), 이메일 주소
|
||||||
|
선택 항목: 소속, 자기소개
|
||||||
|
|
||||||
|
2. 개인정보의 수집 및 이용 목적
|
||||||
|
회원 관리
|
||||||
|
회원 식별 및 인증
|
||||||
|
부정 이용 방지 및 비인가 사용 방지
|
||||||
|
서비스 이용에 따른 문의 사항 처리
|
||||||
|
서비스 제공
|
||||||
|
게임 방 생성 및 참여 등 기본 서비스 제공
|
||||||
|
통계 및 순위 제공 등 부가 서비스 제공
|
||||||
|
고객 지원 및 공지사항 전달
|
||||||
|
서비스 관련 중요한 공지사항 전달
|
||||||
|
이용자 문의 및 불만 처리
|
||||||
|
|
||||||
|
3. 개인정보의 보유 및 이용 기간
|
||||||
|
회원 탈퇴 시: 수집된 모든 개인정보는 회원 탈퇴 즉시 파기합니다.
|
||||||
|
관련 법령에 따른 보관: 전자상거래 등에서의 소비자 보호에 관한 법률 등 관계 법령의 규정에 따라 일정 기간 보관이 필요한 경우 해당 기간 동안 보관합니다.
|
||||||
|
계약 또는 청약 철회 등에 관한 기록: 5년 보관
|
||||||
|
대금 결제 및 재화 등의 공급에 관한 기록: 5년 보관
|
||||||
|
소비자의 불만 또는 분쟁 처리에 관한 기록: 3년 보관
|
||||||
|
|
||||||
|
4. 개인정보의 파기 절차 및 방법
|
||||||
|
파기 절차
|
||||||
|
회원 탈퇴 요청 또는 개인정보 수집 및 이용 목적이 달성된 후 지체 없이 해당 정보를 파기합니다.
|
||||||
|
파기 방법
|
||||||
|
전자적 파일 형태: 복구 및 재생이 불가능한 방법으로 영구 삭제
|
||||||
|
종이 문서 형태: 분쇄하거나 소각
|
||||||
|
|
||||||
|
5. 이용자의 권리 및 행사 방법
|
||||||
|
이용자는 언제든지 자신의 개인정보에 대해 열람, 수정, 삭제, 처리 정지를 요구할 수 있습니다.
|
||||||
|
회원 탈퇴를 원하시는 경우, 서비스 내의 "회원 탈퇴" 기능을 이용하시거나 고객센터를 통해 요청하실 수 있습니다.
|
||||||
|
|
||||||
|
6. 동의를 거부할 권리 및 거부 시 불이익
|
||||||
|
이용자는 개인정보 수집 및 이용에 대한 동의를 거부할 권리가 있습니다.
|
||||||
|
그러나 필수 항목에 대한 동의를 거부하실 경우 서비스 이용이 제한될 수 있습니다.
|
||||||
|
|
||||||
|
7. 개인정보 보호책임자
|
||||||
|
연락처: eld_yeojh@naver.com
|
||||||
|
|
||||||
|
8. 개인정보의 안전성 확보 조치
|
||||||
|
회사는 개인정보의 안전한 처리를 위하여 기술적, 관리적 보호조치를 시행하고 있습니다.
|
||||||
|
개인정보의 암호화
|
||||||
|
해킹 등에 대비한 대책
|
||||||
|
접근 통제 장치의 설치 및 운영
|
||||||
|
''',
|
||||||
|
style: TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
),
|
),
|
||||||
child: const Text('확인'),
|
onPressed: () => Navigator.pop(ctx, false),
|
||||||
onPressed: () {
|
child: Text('거부'),
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
|
TextButton(
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
onPressed: () => Navigator.pop(ctx, true),
|
||||||
|
child: Text('동의'),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
@ -147,52 +409,111 @@ class _LoginPageState extends State<LoginPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (F) 간단 Alert
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
void _showAlert(String title, String message) {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (_) => AlertDialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
title: Text(title, style: const TextStyle(color: Colors.black)),
|
||||||
|
content: Text(message, style: const TextStyle(color: Colors.black)),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
onPressed: () => Navigator.pop(context),
|
||||||
|
child: const Text('확인'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// (G) 화면
|
||||||
|
// ─────────────────────────────────────────
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return WillPopScope(
|
||||||
|
onWillPop: _onWillPop,
|
||||||
|
child: Stack(
|
||||||
|
children: [
|
||||||
|
Scaffold(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
|
|
||||||
|
// 상단 AppBar
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: const Text('ALL SCORE', style: TextStyle(color: Colors.white)),
|
title: const Text('ALL SCORE', style: TextStyle(color: Colors.white)),
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
),
|
),
|
||||||
body: Padding(
|
|
||||||
padding: const EdgeInsets.all(16.0),
|
// 전체 세로 레이아웃
|
||||||
|
body: Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
// (1) 중앙 영역 → 로그인 UI
|
||||||
|
Expanded(
|
||||||
|
child: Center(
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// 1. 올스코어 로그인 (ID/PW)
|
||||||
|
// ─────────────────────────────────────────
|
||||||
const Text(
|
const Text(
|
||||||
'로그인',
|
'올스코어 로그인',
|
||||||
style: TextStyle(
|
style: TextStyle(fontSize: 24, fontWeight: FontWeight.bold, color: Colors.black),
|
||||||
fontSize: 24,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: Colors.black,
|
|
||||||
),
|
),
|
||||||
),
|
const SizedBox(height: 16),
|
||||||
const SizedBox(height: 32),
|
|
||||||
TextField(
|
// (A) 아이디 입력
|
||||||
|
SizedBox(
|
||||||
|
width: 300,
|
||||||
|
child: TextField(
|
||||||
controller: idController,
|
controller: idController,
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'ID',
|
labelText: 'ID',
|
||||||
labelStyle: const TextStyle(color: Colors.black),
|
border: OutlineInputBorder(
|
||||||
border: const OutlineInputBorder(),
|
borderSide: const BorderSide(color: Colors.black),
|
||||||
focusedBorder: const OutlineInputBorder(
|
borderRadius: BorderRadius.circular(8),
|
||||||
borderSide: BorderSide(color: Colors.black, width: 2.0),
|
),
|
||||||
|
enabledBorder: OutlineInputBorder(
|
||||||
|
borderSide: const BorderSide(color: Colors.black),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
),
|
||||||
TextField(
|
const SizedBox(height: 12),
|
||||||
|
|
||||||
|
// (B) 비밀번호 입력
|
||||||
|
SizedBox(
|
||||||
|
width: 300,
|
||||||
|
child: TextField(
|
||||||
controller: passwordController,
|
controller: passwordController,
|
||||||
obscureText: true,
|
obscureText: true,
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'PW',
|
labelText: 'PW',
|
||||||
labelStyle: const TextStyle(color: Colors.black),
|
border: OutlineInputBorder(
|
||||||
border: const OutlineInputBorder(),
|
borderSide: const BorderSide(color: Colors.black),
|
||||||
focusedBorder: const OutlineInputBorder(
|
borderRadius: BorderRadius.circular(8),
|
||||||
borderSide: BorderSide(color: Colors.black, width: 2.0),
|
),
|
||||||
|
enabledBorder: OutlineInputBorder(
|
||||||
|
borderSide: const BorderSide(color: Colors.black),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// (C) 로그인 에러
|
||||||
if (loginErrorMessage.isNotEmpty)
|
if (loginErrorMessage.isNotEmpty)
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(top: 8.0),
|
padding: const EdgeInsets.only(top: 8.0),
|
||||||
@ -201,70 +522,174 @@ class _LoginPageState extends State<LoginPage> {
|
|||||||
style: const TextStyle(color: Colors.red),
|
style: const TextStyle(color: Colors.red),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
const SizedBox(height: 8),
|
||||||
|
|
||||||
|
// (D) 자동로그인 체크박스
|
||||||
|
SizedBox(
|
||||||
|
width: 300,
|
||||||
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
Checkbox(
|
Checkbox(
|
||||||
value: autoLogin,
|
value: autoLogin,
|
||||||
onChanged: (bool? value) {
|
onChanged: (val) {
|
||||||
setState(() {
|
setState(() {
|
||||||
autoLogin = value ?? false;
|
autoLogin = val ?? false;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
const Text('자동로그인', style: TextStyle(color: Colors.black)),
|
const Text('자동로그인', style: TextStyle(color: Colors.black)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
|
||||||
ElevatedButton(
|
|
||||||
onPressed: _login,
|
|
||||||
style: ElevatedButton.styleFrom(
|
|
||||||
backgroundColor: Colors.black,
|
|
||||||
foregroundColor: Colors.white,
|
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// (E) 로그인 버튼
|
||||||
|
SizedBox(
|
||||||
|
width: 300,
|
||||||
|
child: ElevatedButton(
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.black,
|
||||||
|
side: const BorderSide(color: Colors.black),
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
|
),
|
||||||
|
onPressed: _loginWithIdPw,
|
||||||
child: const Text('로그인'),
|
child: const Text('로그인'),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
),
|
||||||
|
|
||||||
|
// (F) ID/PW 찾기, 회원가입
|
||||||
|
const SizedBox(height: 8),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
Navigator.push(
|
Navigator.push(context, MaterialPageRoute(builder: (_) => const IdFindingPage()));
|
||||||
context,
|
|
||||||
MaterialPageRoute(builder: (context) => const IdFindingPage()),
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
child: const Text('ID 찾기', style: TextStyle(color: Colors.black)),
|
child: const Text('ID 찾기', style: TextStyle(color: Colors.black)),
|
||||||
),
|
),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
Navigator.push(
|
Navigator.push(context, MaterialPageRoute(builder: (_) => const PwFindingPage()));
|
||||||
context,
|
|
||||||
MaterialPageRoute(builder: (context) => const PwFindingPage()),
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
child: const Text('PW 찾기', style: TextStyle(color: Colors.black)),
|
child: const Text('PW 찾기', style: TextStyle(color: Colors.black)),
|
||||||
),
|
),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
Navigator.push(
|
Navigator.push(context, MaterialPageRoute(builder: (_) => const SignUpPage()));
|
||||||
context,
|
|
||||||
MaterialPageRoute(builder: (context) => const SignUpPage()),
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
child: const Text('회원가입', style: TextStyle(color: Colors.black)),
|
child: const Text('회원가입', style: TextStyle(color: Colors.black)),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
const Divider(height: 1, color: Colors.black),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
// 2. 구글 로그인 / 회원가입
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
const Text(
|
||||||
|
'구글 계정',
|
||||||
|
style: TextStyle(fontSize: 20, color: Colors.black, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
// 광고 영역
|
|
||||||
Container(
|
// (a) 구글 로그인
|
||||||
height: 50,
|
SizedBox(
|
||||||
color: Colors.grey[300],
|
width: 300,
|
||||||
child: const Center(child: Text('광고 영역', style: TextStyle(color: Colors.black))),
|
child: ElevatedButton.icon(
|
||||||
|
icon: Container(
|
||||||
|
width: 24,
|
||||||
|
height: 24,
|
||||||
|
decoration: const BoxDecoration(
|
||||||
|
image: DecorationImage(
|
||||||
|
image: AssetImage('assets/images/icons8-google-logo-48.png'),
|
||||||
|
fit: BoxFit.contain,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
label: const Text(
|
||||||
|
'Google 로그인',
|
||||||
|
style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.black,
|
||||||
|
side: const BorderSide(color: Colors.black),
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 24),
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
|
),
|
||||||
|
onPressed: _googleLogin,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 12),
|
||||||
|
|
||||||
|
// (b) 구글 회원가입
|
||||||
|
SizedBox(
|
||||||
|
width: 300,
|
||||||
|
child: ElevatedButton.icon(
|
||||||
|
icon: Container(
|
||||||
|
width: 24,
|
||||||
|
height: 24,
|
||||||
|
decoration: const BoxDecoration(
|
||||||
|
image: DecorationImage(
|
||||||
|
image: AssetImage('assets/images/icons8-google-logo-48.png'),
|
||||||
|
fit: BoxFit.contain,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
label: const Text(
|
||||||
|
'Google 회원가입',
|
||||||
|
style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.black,
|
||||||
|
side: const BorderSide(color: Colors.black),
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 24),
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
|
),
|
||||||
|
onPressed: _googleSignUp,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// (2) 하단 광고 영역
|
||||||
|
if (_isBannerReady && _bannerAd != null)
|
||||||
|
Container(
|
||||||
|
width: _bannerAd!.size.width.toDouble(),
|
||||||
|
height: _bannerAd!.size.height.toDouble(),
|
||||||
|
alignment: Alignment.center,
|
||||||
|
child: AdWidget(ad: _bannerAd!),
|
||||||
|
)
|
||||||
|
else
|
||||||
|
Container(
|
||||||
|
width: 300,
|
||||||
|
height: 50,
|
||||||
|
color: Colors.grey.shade400,
|
||||||
|
alignment: Alignment.center,
|
||||||
|
child: const Text(
|
||||||
|
'광고 로딩중',
|
||||||
|
style: TextStyle(color: Colors.black),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// (2) 로딩 중일 때 오버레이 표시
|
||||||
|
if (_isLoading)
|
||||||
|
Container(
|
||||||
|
width: double.infinity,
|
||||||
|
height: double.infinity,
|
||||||
|
color: Colors.black54, // 반투명 배경
|
||||||
|
alignment: Alignment.center,
|
||||||
|
child: const CircularProgressIndicator(color: Colors.white),
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -6,6 +6,12 @@ import 'login_page.dart'; // 로그인 페이지 임포트 추가
|
|||||||
import 'signup_page.dart'; // 회원가입 페이지 임포트 추가
|
import 'signup_page.dart'; // 회원가입 페이지 임포트 추가
|
||||||
import 'id_finding_page.dart'; // ID 찾기 페이지 임포트 추가
|
import 'id_finding_page.dart'; // ID 찾기 페이지 임포트 추가
|
||||||
|
|
||||||
|
// 모바일 광고
|
||||||
|
import 'package:google_mobile_ads/google_mobile_ads.dart';
|
||||||
|
|
||||||
|
// 설정
|
||||||
|
import '../../config/config.dart';
|
||||||
|
|
||||||
class PwFindingPage extends StatefulWidget {
|
class PwFindingPage extends StatefulWidget {
|
||||||
const PwFindingPage({Key? key}) : super(key: key);
|
const PwFindingPage({Key? key}) : super(key: key);
|
||||||
|
|
||||||
@ -19,6 +25,11 @@ class _PwFindingPageState extends State<PwFindingPage> {
|
|||||||
String emailErrorMessage = ''; // 이메일 오류 메시지
|
String emailErrorMessage = ''; // 이메일 오류 메시지
|
||||||
String idErrorMessage = ''; // ID 오류 메시지
|
String idErrorMessage = ''; // ID 오류 메시지
|
||||||
|
|
||||||
|
/// (1) 광고 배너 관련 변수
|
||||||
|
BannerAd? _bannerAd;
|
||||||
|
bool _isBannerReady = false; // 광고 로드 완료 여부
|
||||||
|
String adUnitId = Config.testAdUnitId;
|
||||||
|
|
||||||
Future<void> _findPassword(String id, String email) async {
|
Future<void> _findPassword(String id, String email) async {
|
||||||
// PW 찾기 요청 처리
|
// PW 찾기 요청 처리
|
||||||
print('PW 찾기 요청: ID: $id, 이메일: $email'); // 요청 시 출력
|
print('PW 찾기 요청: ID: $id, 이메일: $email'); // 요청 시 출력
|
||||||
@ -113,6 +124,36 @@ class _PwFindingPageState extends State<PwFindingPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_initBannerAd();
|
||||||
|
}
|
||||||
|
|
||||||
|
void _initBannerAd() {
|
||||||
|
_bannerAd = BannerAd(
|
||||||
|
// 실제/테스트 배너 광고 단위 ID
|
||||||
|
adUnitId: adUnitId,
|
||||||
|
size: AdSize.banner,
|
||||||
|
request: const AdRequest(),
|
||||||
|
listener: BannerAdListener(
|
||||||
|
onAdLoaded: (ad) {
|
||||||
|
setState(() => _isBannerReady = true);
|
||||||
|
},
|
||||||
|
onAdFailedToLoad: (ad, error) {
|
||||||
|
ad.dispose();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
_bannerAd?.load();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_bannerAd?.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
@ -213,6 +254,16 @@ class _PwFindingPageState extends State<PwFindingPage> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// (3) 하단 광고 영역
|
||||||
|
bottomNavigationBar: _isBannerReady && _bannerAd != null
|
||||||
|
? Container(
|
||||||
|
color: Colors.white,
|
||||||
|
width: _bannerAd!.size.width.toDouble(),
|
||||||
|
height: _bannerAd!.size.height.toDouble(),
|
||||||
|
child: AdWidget(ad: _bannerAd!),
|
||||||
|
)
|
||||||
|
: SizedBox.shrink(), // 로딩 전엔 빈 공간 or 원하는 위젯
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -21,7 +21,7 @@ class _SignUpPageState extends State<SignUpPage> {
|
|||||||
|
|
||||||
// 유효성 검사
|
// 유효성 검사
|
||||||
bool _isUsernameValid(String username) => RegExp(r'^(?![0-9])[A-Za-z0-9]{6,20}$').hasMatch(username);
|
bool _isUsernameValid(String username) => RegExp(r'^(?![0-9])[A-Za-z0-9]{6,20}$').hasMatch(username);
|
||||||
bool _isPasswordValidPattern(String password) => RegExp(r'^(?=.*[A-Za-z])(?=.*\d)[A-Za-z\d]{8,20}$').hasMatch(password);
|
bool _isPasswordValidPattern(String password) => RegExp(r"""^(?=.*[A-Za-z])(?=.*\d)[A-Za-z\d!@#$%^&*()_\-+=~`{}\[\]|\\:;\"'<>,.?/]{8,20}$""").hasMatch(password);
|
||||||
bool _isEmailValid(String email) => RegExp(r'^[\w-\.]+@([\w-]+\.)+[\w-]{2,4}$').hasMatch(email);
|
bool _isEmailValid(String email) => RegExp(r'^[\w-\.]+@([\w-]+\.)+[\w-]{2,4}$').hasMatch(email);
|
||||||
bool _isNicknameValid(String nickname) => RegExp(r'^[A-Za-z가-힣0-9]{2,20}$').hasMatch(nickname);
|
bool _isNicknameValid(String nickname) => RegExp(r'^[A-Za-z가-힣0-9]{2,20}$').hasMatch(nickname);
|
||||||
|
|
||||||
|
@ -1,48 +1,325 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
import 'main_page.dart';
|
import '../../plugins/api.dart';
|
||||||
|
import '../../dialogs/response_dialog.dart';
|
||||||
|
import '../../dialogs/room_setting_finish_dialog.dart';
|
||||||
|
import '../../dialogs/user_info_finish_dialog.dart';
|
||||||
|
|
||||||
class FinishPrivatePage extends StatelessWidget {
|
class FinishPrivatePage extends StatefulWidget {
|
||||||
final int roomSeq;
|
final int roomSeq;
|
||||||
|
final bool fromPlayingPage; // 만약 대기/진행중에서 넘어온 경우 => 뒤로가기 시 메인으로
|
||||||
|
|
||||||
const FinishPrivatePage({
|
const FinishPrivatePage({
|
||||||
Key? key,
|
Key? key,
|
||||||
required this.roomSeq,
|
required this.roomSeq,
|
||||||
|
this.fromPlayingPage = false,
|
||||||
}) : super(key: key);
|
}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
State<FinishPrivatePage> createState() => _FinishPrivatePageState();
|
||||||
// 간단한 종료 안내 화면
|
}
|
||||||
return Scaffold(
|
|
||||||
backgroundColor: Colors.white,
|
class _FinishPrivatePageState extends State<FinishPrivatePage> {
|
||||||
appBar: AppBar(
|
bool _isLoading = true;
|
||||||
title: const Text('게임 종료 (개인전)', style: TextStyle(color: Colors.white)),
|
|
||||||
backgroundColor: Colors.black,
|
Map<String, dynamic> _roomInfo = {}; // 서버에서 받은 room_info
|
||||||
leading: IconButton(
|
// 전체 user_info Map
|
||||||
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
// userSeq → { user_seq, nickname, participant_type, score, ... }
|
||||||
onPressed: () {
|
Map<String, dynamic> _userMap = {};
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
},
|
// 리스트로 만든 (관리자 제외) 참가자 목록 (점수 내림차순)
|
||||||
),
|
List<Map<String, dynamic>> _playerList = [];
|
||||||
),
|
// 별도 사회자(ADMIN) 목록 (개인전이라 1명이거나 없을 수 있음)
|
||||||
body: Center(
|
List<Map<String, dynamic>> _adminList = [];
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
String _roomTitle = '';
|
||||||
|
DateTime? _startDt;
|
||||||
|
DateTime? _endDt;
|
||||||
|
int _masterUserSeq = 0; // 방장 user_seq (ADMIN과는 별개)
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_fetchFinishRoomInfo();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// (A) 서버에서 종료된 방 정보를 가져옴
|
||||||
|
Future<void> _fetchFinishRoomInfo() async {
|
||||||
|
setState(() => _isLoading = true);
|
||||||
|
try {
|
||||||
|
final body = {"room_seq": "${widget.roomSeq}"};
|
||||||
|
final response = await Api.serverRequest(
|
||||||
|
uri: '/room/score/get/finish/room/info',
|
||||||
|
body: body,
|
||||||
|
);
|
||||||
|
if (response['result'] == 'OK') {
|
||||||
|
final resp = response['response'] ?? {};
|
||||||
|
if (resp['result'] == 'OK') {
|
||||||
|
final data = resp['data'] ?? {};
|
||||||
|
|
||||||
|
final rInfo = data['room_info'] ?? {};
|
||||||
|
final uInfo = data['user_info'] ?? {};
|
||||||
|
|
||||||
|
final rTitle = (rInfo['room_title'] ?? '') as String;
|
||||||
|
final mSeq = (rInfo['master_user_seq'] ?? 0) as int;
|
||||||
|
final sdt = rInfo['start_dt'];
|
||||||
|
final edt = rInfo['end_dt'];
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_roomInfo = rInfo;
|
||||||
|
_userMap = uInfo;
|
||||||
|
_roomTitle = rTitle.isNotEmpty ? rTitle : '종료된 방(개인전)';
|
||||||
|
_masterUserSeq = mSeq;
|
||||||
|
|
||||||
|
if (sdt != null && sdt is String && sdt.contains('T')) {
|
||||||
|
_startDt = DateTime.tryParse(sdt);
|
||||||
|
}
|
||||||
|
if (edt != null && edt is String && edt.contains('T')) {
|
||||||
|
_endDt = DateTime.tryParse(edt);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// userInfo -> List 변환
|
||||||
|
final List<Map<String, dynamic>> tempList = [];
|
||||||
|
uInfo.forEach((_, val) {
|
||||||
|
// val: { user_seq, participant_type, nickname, score, ... }
|
||||||
|
tempList.add(Map<String, dynamic>.from(val));
|
||||||
|
});
|
||||||
|
|
||||||
|
// (1) 사회자(ADMIN) 분리
|
||||||
|
final adminList = tempList.where((u) {
|
||||||
|
final pType = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
|
return pType == 'ADMIN';
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
// (2) 플레이어 목록 (ADMIN 제외) & 점수 내림차순
|
||||||
|
final playerList = tempList.where((u) {
|
||||||
|
final pType = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
|
return pType != 'ADMIN';
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
// 점수 내림차순 정렬
|
||||||
|
playerList.sort((a, b) {
|
||||||
|
final sa = (a['score'] ?? 0) as int;
|
||||||
|
final sb = (b['score'] ?? 0) as int;
|
||||||
|
return sb.compareTo(sa); // 내림차순
|
||||||
|
});
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_adminList = adminList;
|
||||||
|
_playerList = playerList;
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
final msgTitle = resp['response_info']?['msg_title'] ?? '오류';
|
||||||
|
final msgContent = resp['response_info']?['msg_content'] ?? '데이터 조회 실패';
|
||||||
|
showResponseDialog(context, msgTitle, msgContent);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showResponseDialog(context, '실패', '서버 통신 오류');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
showResponseDialog(context, '오류', '$e');
|
||||||
|
} finally {
|
||||||
|
setState(() => _isLoading = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// (B) 뒤로가기
|
||||||
|
Future<bool> _onWillPop() async {
|
||||||
|
if (widget.fromPlayingPage) {
|
||||||
|
// 진행중/대기중 등에서 넘어왔다면 => 메인으로
|
||||||
|
Navigator.popUntil(context, (route) => route.isFirst);
|
||||||
|
} else {
|
||||||
|
// 검색 등에서 왔으면 => 한 단계 pop
|
||||||
|
Navigator.pop(context);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// (C) 방 정보보기 모달 (읽기 전용)
|
||||||
|
Future<void> _openRoomSettingDialog() async {
|
||||||
|
if (_roomInfo.isEmpty) return;
|
||||||
|
await showDialog(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) => RoomSettingFinishDialog(roomInfo: _roomInfo),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// (D) 게임 진행 시간 표시
|
||||||
|
Widget _buildGameTimeWidget() {
|
||||||
|
if (_startDt == null || _endDt == null) {
|
||||||
|
return const Text('게임 진행 시간: 00:00', style: TextStyle(fontSize: 14));
|
||||||
|
}
|
||||||
|
final dur = _endDt!.difference(_startDt!);
|
||||||
|
final hh = dur.inHours.toString().padLeft(2, '0');
|
||||||
|
final mm = dur.inMinutes.remainder(60).toString().padLeft(2, '0');
|
||||||
|
return Text('게임 진행 시간: $hh:$mm', style: const TextStyle(fontSize: 14));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// (E) 플레이어 목록 표시 (점수 내림차순)
|
||||||
|
/// - 1등/2등/3등 금/은/동 메달
|
||||||
|
Widget _buildPlayerItem(Map<String, dynamic> user, int index) {
|
||||||
|
final score = (user['score'] ?? 0) as int;
|
||||||
|
final nickname = user['nickname'] ?? '유저';
|
||||||
|
final profileImg = user['profile_img'] ?? '';
|
||||||
|
final userSeq = user['user_seq'] ?? 0;
|
||||||
|
|
||||||
|
Widget medal = const SizedBox();
|
||||||
|
if (index == 0) {
|
||||||
|
medal = const Text('🥇 ', style: TextStyle(fontSize: 16));
|
||||||
|
} else if (index == 1) {
|
||||||
|
medal = const Text('🥈 ', style: TextStyle(fontSize: 16));
|
||||||
|
} else if (index == 2) {
|
||||||
|
medal = const Text('🥉 ', style: TextStyle(fontSize: 16));
|
||||||
|
}
|
||||||
|
|
||||||
|
return GestureDetector(
|
||||||
|
onTap: () => _onTapUser(user),
|
||||||
|
child: Container(
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 4),
|
||||||
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
const Text('게임이 종료되었습니다.', style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold)),
|
medal,
|
||||||
const SizedBox(height: 16),
|
// 프로필
|
||||||
ElevatedButton(
|
Container(
|
||||||
onPressed: () {
|
width: 36, height: 36,
|
||||||
// 메인 페이지로 이동
|
decoration: BoxDecoration(
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
shape: BoxShape.circle,
|
||||||
},
|
border: Border.all(color: Colors.black54),
|
||||||
style: ElevatedButton.styleFrom(backgroundColor: Colors.black),
|
|
||||||
child: const Text('메인으로', style: TextStyle(color: Colors.white)),
|
|
||||||
),
|
),
|
||||||
|
child: ClipOval(
|
||||||
|
child: (profileImg.isNotEmpty)
|
||||||
|
? Image.network(
|
||||||
|
'https://eldsoft.com:8097/images$profileImg',
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
errorBuilder: (_, __, ___) => const Center(child: Text('ERR')),
|
||||||
|
)
|
||||||
|
: const Center(child: Text('No\nImg', textAlign: TextAlign.center, style: TextStyle(fontSize: 10))),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
Expanded(
|
||||||
|
child: Text(nickname, style: const TextStyle(fontSize: 14, fontWeight: FontWeight.bold)),
|
||||||
|
),
|
||||||
|
Text('$score 점', style: const TextStyle(fontSize: 14)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// 사회자 목록 (보통 1명 예상)
|
||||||
|
Widget _buildAdminItem(Map<String, dynamic> admin) {
|
||||||
|
final nickname = admin['nickname'] ?? '사회자';
|
||||||
|
final profileImg = admin['profile_img'] ?? '';
|
||||||
|
return GestureDetector(
|
||||||
|
onTap: () => _onTapUser(admin),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
width: 36, height: 36,
|
||||||
|
margin: const EdgeInsets.only(right: 8),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border: Border.all(color: Colors.deepPurple),
|
||||||
|
),
|
||||||
|
child: ClipOval(
|
||||||
|
child: (profileImg.isNotEmpty)
|
||||||
|
? Image.network(
|
||||||
|
'https://eldsoft.com:8097/images$profileImg',
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
errorBuilder: (_, __, ___) => const Center(child: Text('ERR')),
|
||||||
|
)
|
||||||
|
: const Center(child: Text('No\nImg', textAlign: TextAlign.center, style: TextStyle(fontSize: 10))),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Text(nickname, style: const TextStyle(fontSize: 14, fontWeight: FontWeight.bold, color: Colors.deepPurple)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 사용자 클릭 -> 새 유저 정보 모달
|
||||||
|
Future<void> _onTapUser(Map<String, dynamic> userData) async {
|
||||||
|
// user_info_finish_dialog.dart (새 모달)
|
||||||
|
await showDialog(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) => UserInfoFinishDialog(userData: userData),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return WillPopScope(
|
||||||
|
onWillPop: _onWillPop,
|
||||||
|
child: Scaffold(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
appBar: AppBar(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
leading: IconButton(
|
||||||
|
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
||||||
|
onPressed: _onWillPop,
|
||||||
|
),
|
||||||
|
title: Text(_roomTitle, style: const TextStyle(color: Colors.white)),
|
||||||
|
),
|
||||||
|
body: _isLoading
|
||||||
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
// (A) 상단: [방 정보 보기] 버튼 + 진행 시간
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
// 방번호는 숨기라 했으므로 제거
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: _openRoomSettingDialog,
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
side: const BorderSide(color: Colors.black, width: 1),
|
||||||
|
),
|
||||||
|
child: const Text('방 정보 보기', style: TextStyle(color: Colors.black)),
|
||||||
|
),
|
||||||
|
_buildGameTimeWidget(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
|
||||||
|
// (B) 사회자 목록
|
||||||
|
if (_adminList.isNotEmpty) ...[
|
||||||
|
Container(
|
||||||
|
width: double.infinity,
|
||||||
|
padding: const EdgeInsets.all(8),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
border: Border.all(color: Colors.deepPurple),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
const Text('사회자: ', style: TextStyle(fontSize: 14, color: Colors.deepPurple)),
|
||||||
|
..._adminList.map(_buildAdminItem),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
],
|
||||||
|
|
||||||
|
// (C) 참가자 목록
|
||||||
|
ListView.builder(
|
||||||
|
primary: false,
|
||||||
|
shrinkWrap: true,
|
||||||
|
itemCount: _playerList.length,
|
||||||
|
itemBuilder: (ctx, i) => _buildPlayerItem(_playerList[i], i),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,47 +1,399 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
import 'main_page.dart';
|
|
||||||
|
|
||||||
class FinishTeamPage extends StatelessWidget {
|
import '../../plugins/api.dart';
|
||||||
|
import '../../dialogs/response_dialog.dart';
|
||||||
|
import '../../dialogs/room_setting_finish_dialog.dart';
|
||||||
|
import '../../dialogs/user_info_finish_dialog.dart';
|
||||||
|
|
||||||
|
class FinishTeamPage extends StatefulWidget {
|
||||||
final int roomSeq;
|
final int roomSeq;
|
||||||
|
final bool fromPlayingPage; // 진행중에서 넘어왔는지 여부
|
||||||
|
|
||||||
const FinishTeamPage({
|
const FinishTeamPage({
|
||||||
Key? key,
|
Key? key,
|
||||||
required this.roomSeq,
|
required this.roomSeq,
|
||||||
|
this.fromPlayingPage = false,
|
||||||
}) : super(key: key);
|
}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
State<FinishTeamPage> createState() => _FinishTeamPageState();
|
||||||
// 간단한 종료 안내
|
}
|
||||||
return Scaffold(
|
|
||||||
backgroundColor: Colors.white,
|
class _FinishTeamPageState extends State<FinishTeamPage> {
|
||||||
appBar: AppBar(
|
bool _isLoading = true;
|
||||||
title: const Text('게임 종료 (팀전)', style: TextStyle(color: Colors.white)),
|
|
||||||
backgroundColor: Colors.black,
|
Map<String, dynamic> _roomInfo = {};
|
||||||
leading: IconButton(
|
Map<String, dynamic> _userMap = {};
|
||||||
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
List<Map<String, dynamic>> _userList = [];
|
||||||
onPressed: () {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
String _roomTitle = '';
|
||||||
},
|
int _masterUserSeq = 0;
|
||||||
|
DateTime? _startDt;
|
||||||
|
DateTime? _endDt;
|
||||||
|
|
||||||
|
// 팀별 [ { user }, { user } ... ]
|
||||||
|
Map<String, List<Map<String, dynamic>>> _teamMap = {};
|
||||||
|
// 팀별 점수
|
||||||
|
Map<String, int> _teamScoreMap = {};
|
||||||
|
|
||||||
|
// 별도 사회자 목록
|
||||||
|
List<Map<String, dynamic>> _adminList = [];
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_fetchFinishRoomInfo();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _fetchFinishRoomInfo() async {
|
||||||
|
setState(() => _isLoading = true);
|
||||||
|
try {
|
||||||
|
final body = {"room_seq": "${widget.roomSeq}"};
|
||||||
|
final response = await Api.serverRequest(
|
||||||
|
uri: '/room/score/get/finish/room/info',
|
||||||
|
body: body,
|
||||||
|
);
|
||||||
|
if (response['result'] == 'OK') {
|
||||||
|
final resp = response['response'] ?? {};
|
||||||
|
if (resp['result'] == 'OK') {
|
||||||
|
final data = resp['data'] ?? {};
|
||||||
|
final rInfo = data['room_info'] ?? {};
|
||||||
|
final uInfo = data['user_info'] ?? {};
|
||||||
|
|
||||||
|
final rTitle = (rInfo['room_title'] ?? '') as String;
|
||||||
|
final mSeq = (rInfo['master_user_seq'] ?? 0) as int;
|
||||||
|
final sdt = rInfo['start_dt'];
|
||||||
|
final edt = rInfo['end_dt'];
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_roomInfo = rInfo;
|
||||||
|
_userMap = uInfo;
|
||||||
|
_roomTitle = rTitle.isNotEmpty ? rTitle : '종료된 팀전';
|
||||||
|
_masterUserSeq = mSeq;
|
||||||
|
|
||||||
|
if (sdt != null && sdt is String && sdt.contains('T')) {
|
||||||
|
_startDt = DateTime.tryParse(sdt);
|
||||||
|
}
|
||||||
|
if (edt != null && edt is String && edt.contains('T')) {
|
||||||
|
_endDt = DateTime.tryParse(edt);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// userList
|
||||||
|
final List<Map<String, dynamic>> tempList = [];
|
||||||
|
uInfo.forEach((_, val) {
|
||||||
|
tempList.add(Map<String, dynamic>.from(val));
|
||||||
|
});
|
||||||
|
|
||||||
|
// (1) 사회자(ADMIN) 분리
|
||||||
|
final adminList = tempList.where((u) {
|
||||||
|
final pType = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
|
return pType == 'ADMIN';
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
// (2) 일반 참가자
|
||||||
|
final players = tempList.where((u) {
|
||||||
|
final pType = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
|
return pType != 'ADMIN';
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
// (3) 팀명별 분류 + 점수 합
|
||||||
|
final Map<String, List<Map<String, dynamic>>> tMap = {};
|
||||||
|
final Map<String, int> tScoreMap = {};
|
||||||
|
|
||||||
|
for (var user in players) {
|
||||||
|
final tName = (user['team_name'] ?? 'WAIT').toString().toUpperCase();
|
||||||
|
if (tName == 'WAIT') continue; // 팀 미배정
|
||||||
|
tMap.putIfAbsent(tName, () => []);
|
||||||
|
tMap[tName]!.add(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 팀별 점수
|
||||||
|
tMap.forEach((team, mems) {
|
||||||
|
int sumScore = 0;
|
||||||
|
// mems 내림차순 정렬
|
||||||
|
mems.sort((a, b) {
|
||||||
|
final sa = (a['score'] ?? 0) as int;
|
||||||
|
final sb = (b['score'] ?? 0) as int;
|
||||||
|
return sb.compareTo(sa);
|
||||||
|
});
|
||||||
|
for (var m in mems) {
|
||||||
|
sumScore += (m['score'] ?? 0) as int;
|
||||||
|
}
|
||||||
|
tScoreMap[team] = sumScore;
|
||||||
|
});
|
||||||
|
|
||||||
|
// (4) 팀들을 점수 순으로 정렬
|
||||||
|
final sortedTeams = tScoreMap.keys.toList();
|
||||||
|
sortedTeams.sort((a, b) => tScoreMap[b]!.compareTo(tScoreMap[a]!));
|
||||||
|
|
||||||
|
// 정렬된 결과를 새 맵에 담음
|
||||||
|
final Map<String, List<Map<String, dynamic>>> finalTeamMap = {};
|
||||||
|
final Map<String, int> finalScoreMap = {};
|
||||||
|
|
||||||
|
for (var t in sortedTeams) {
|
||||||
|
finalTeamMap[t] = tMap[t]!;
|
||||||
|
finalScoreMap[t] = tScoreMap[t]!;
|
||||||
|
}
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_adminList = adminList;
|
||||||
|
_userList = tempList; // 전체 필요하면 보관
|
||||||
|
_teamMap = finalTeamMap;
|
||||||
|
_teamScoreMap = finalScoreMap;
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
final msgTitle = resp['response_info']?['msg_title'] ?? '오류';
|
||||||
|
final msgContent = resp['response_info']?['msg_content'] ?? '데이터 조회 실패';
|
||||||
|
showResponseDialog(context, msgTitle, msgContent);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showResponseDialog(context, '실패', '서버 통신 오류');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
showResponseDialog(context, '오류', '$e');
|
||||||
|
} finally {
|
||||||
|
setState(() => _isLoading = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<bool> _onWillPop() async {
|
||||||
|
if (widget.fromPlayingPage) {
|
||||||
|
Navigator.popUntil(context, (route) => route.isFirst);
|
||||||
|
} else {
|
||||||
|
Navigator.pop(context);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _openRoomSettingDialog() async {
|
||||||
|
if (_roomInfo.isEmpty) return;
|
||||||
|
await showDialog(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) => RoomSettingFinishDialog(roomInfo: _roomInfo),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildGameTimeWidget() {
|
||||||
|
if (_startDt == null || _endDt == null) {
|
||||||
|
return const Text('게임 진행 시간: 00:00', style: TextStyle(fontSize: 14));
|
||||||
|
}
|
||||||
|
final dur = _endDt!.difference(_startDt!);
|
||||||
|
final hh = dur.inHours.toString().padLeft(2, '0');
|
||||||
|
final mm = dur.inMinutes.remainder(60).toString().padLeft(2, '0');
|
||||||
|
return Text('게임 진행 시간: $hh:$mm', style: const TextStyle(fontSize: 14));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 팀 박스 (팀별 점수 표시 + 1등2등3등 금/은/동 배경)
|
||||||
|
Widget _buildTeamBox(String teamName, int index) {
|
||||||
|
final members = _teamMap[teamName] ?? [];
|
||||||
|
final tScore = _teamScoreMap[teamName] ?? 0;
|
||||||
|
|
||||||
|
// 1등/2등/3등 팀 -> 배경색
|
||||||
|
Color bgColor = Colors.white;
|
||||||
|
if (index == 0) {
|
||||||
|
bgColor = const Color(0xFFFFF9C4); // 약간 금색 계열 예: amber.shade100
|
||||||
|
} else if (index == 1) {
|
||||||
|
bgColor = const Color(0xFFE0E0E0); // 은색(회색) 계열
|
||||||
|
} else if (index == 2) {
|
||||||
|
bgColor = const Color(0xFFFFE0B2); // 동색(주황~갈색 계열)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Container(
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: bgColor,
|
||||||
|
border: Border.all(color: Colors.black),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
),
|
),
|
||||||
),
|
|
||||||
body: Center(
|
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
const Text('게임이 종료되었습니다.', style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold)),
|
// 상단 바
|
||||||
const SizedBox(height: 16),
|
Container(
|
||||||
ElevatedButton(
|
color: Colors.black,
|
||||||
onPressed: () {
|
width: double.infinity,
|
||||||
// 메인페이지
|
padding: const EdgeInsets.all(8),
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
child: Center(
|
||||||
},
|
child: Text(
|
||||||
style: ElevatedButton.styleFrom(backgroundColor: Colors.black),
|
'$teamName 팀 (점수: $tScore)',
|
||||||
child: const Text('메인으로', style: TextStyle(color: Colors.white)),
|
style: const TextStyle(color: Colors.white, fontWeight: FontWeight.bold),
|
||||||
),
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// 팀 멤버
|
||||||
|
Container(
|
||||||
|
padding: const EdgeInsets.all(8),
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
scrollDirection: Axis.horizontal,
|
||||||
|
child: Row(
|
||||||
|
children: members.map((m) => _buildTeamMember(m)).toList(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 팀 멤버 표시
|
||||||
|
Widget _buildTeamMember(Map<String, dynamic> user) {
|
||||||
|
final score = (user['score'] ?? 0) as int;
|
||||||
|
final nickname = user['nickname'] ?? '유저';
|
||||||
|
final profileImg = user['profile_img'] ?? '';
|
||||||
|
|
||||||
|
return GestureDetector(
|
||||||
|
onTap: () => _onTapUser(user),
|
||||||
|
child: Container(
|
||||||
|
width: 60,
|
||||||
|
margin: const EdgeInsets.only(right: 8),
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
Text('$score', style: const TextStyle(fontSize: 16, fontWeight: FontWeight.bold)),
|
||||||
|
const SizedBox(height: 2),
|
||||||
|
Container(
|
||||||
|
width: 30, height: 30,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border: Border.all(color: Colors.black),
|
||||||
|
),
|
||||||
|
child: ClipOval(
|
||||||
|
child: Image.network(
|
||||||
|
'https://eldsoft.com:8097/images$profileImg',
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
errorBuilder: (ctx, err, st) => const Center(
|
||||||
|
child: Text('ERR', style: TextStyle(fontSize: 8)),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 2),
|
||||||
|
Text(nickname, style: const TextStyle(fontSize: 11), overflow: TextOverflow.ellipsis),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// 팀전 사회자 목록 (ADMIN)
|
||||||
|
Widget _buildAdminList() {
|
||||||
|
final adminList = _adminList;
|
||||||
|
if (adminList.isEmpty) return const SizedBox();
|
||||||
|
|
||||||
|
return Container(
|
||||||
|
width: double.infinity,
|
||||||
|
margin: const EdgeInsets.only(top: 16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.white,
|
||||||
|
border: Border.all(color: Colors.deepPurple),
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
color: Colors.black,
|
||||||
|
width: double.infinity,
|
||||||
|
padding: const EdgeInsets.all(8),
|
||||||
|
child: const Center(
|
||||||
|
child: Text('사회자', style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold)),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ListView.builder(
|
||||||
|
shrinkWrap: true,
|
||||||
|
primary: false,
|
||||||
|
itemCount: adminList.length,
|
||||||
|
itemBuilder: (ctx, i) {
|
||||||
|
final user = adminList[i];
|
||||||
|
return _buildAdminItem(user);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildAdminItem(Map<String, dynamic> user) {
|
||||||
|
final nickname = user['nickname'] ?? '사회자';
|
||||||
|
final profileImg = user['profile_img'] ?? '';
|
||||||
|
|
||||||
|
return ListTile(
|
||||||
|
onTap: () => _onTapUser(user),
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
backgroundImage: (profileImg.isNotEmpty)
|
||||||
|
? NetworkImage('https://eldsoft.com:8097/images$profileImg')
|
||||||
|
: null,
|
||||||
|
child: (profileImg.isEmpty)
|
||||||
|
? const Text('NoImg', style: TextStyle(fontSize: 10))
|
||||||
|
: null,
|
||||||
|
),
|
||||||
|
title: Text(nickname),
|
||||||
|
subtitle: const Text('사회자'),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _onTapUser(Map<String, dynamic> userData) async {
|
||||||
|
// 새 유저 정보 모달
|
||||||
|
await showDialog(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) => UserInfoFinishDialog(userData: userData),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final teamNames = _teamMap.keys.toList();
|
||||||
|
// 이미 점수순으로 정렬됨
|
||||||
|
|
||||||
|
return WillPopScope(
|
||||||
|
onWillPop: _onWillPop,
|
||||||
|
child: Scaffold(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
appBar: AppBar(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
leading: IconButton(
|
||||||
|
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
||||||
|
onPressed: _onWillPop,
|
||||||
|
),
|
||||||
|
title: Text(_roomTitle, style: const TextStyle(color: Colors.white)),
|
||||||
|
),
|
||||||
|
body: _isLoading
|
||||||
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
// 방 정보 + 시간
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: _openRoomSettingDialog,
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
side: const BorderSide(color: Colors.black),
|
||||||
|
),
|
||||||
|
child: const Text('방 정보 보기', style: TextStyle(color: Colors.black)),
|
||||||
|
),
|
||||||
|
_buildGameTimeWidget(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
|
||||||
|
// 팀 박스들
|
||||||
|
for (int i = 0; i < teamNames.length; i++)
|
||||||
|
_buildTeamBox(teamNames[i], i),
|
||||||
|
|
||||||
|
// 사회자 목록
|
||||||
|
_buildAdminList(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,24 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:firebase_database/firebase_database.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
import 'package:google_mobile_ads/google_mobile_ads.dart'; // ★ AdMob 패키지
|
||||||
|
// 기타 import...
|
||||||
|
|
||||||
import '../../dialogs/settings_dialog.dart';
|
import '../../dialogs/settings_dialog.dart';
|
||||||
import 'create_room_page.dart';
|
import 'create_room_page.dart';
|
||||||
|
|
||||||
// 새로 추가할 페이지들
|
|
||||||
import 'room_search_home_page.dart';
|
import 'room_search_home_page.dart';
|
||||||
|
|
||||||
|
// 진행중 방
|
||||||
|
import 'playing_private_page.dart';
|
||||||
|
import 'playing_team_page.dart';
|
||||||
|
|
||||||
|
// 임시: 서버 API & 모달
|
||||||
|
import '../../plugins/api.dart';
|
||||||
|
import '../../dialogs/response_dialog.dart';
|
||||||
|
|
||||||
|
// 뒤로가기
|
||||||
|
import 'package:fluttertoast/fluttertoast.dart'; // 뒤로가기 안내 문구에 Toast 등 사용
|
||||||
|
|
||||||
class MainPage extends StatefulWidget {
|
class MainPage extends StatefulWidget {
|
||||||
const MainPage({Key? key}) : super(key: key);
|
const MainPage({Key? key}) : super(key: key);
|
||||||
|
|
||||||
@ -15,25 +27,151 @@ class MainPage extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _MainPageState extends State<MainPage> {
|
class _MainPageState extends State<MainPage> {
|
||||||
bool _isBackButtonVisible = false; // 뒤로가기 버튼 상태
|
/// (1) 광고 배너 관련 변수
|
||||||
|
BannerAd? _bannerAd;
|
||||||
|
bool _isBannerReady = false; // 광고 로드 완료 여부
|
||||||
|
|
||||||
|
// 뒤로가기 처리
|
||||||
|
DateTime? _lastPressedTime;
|
||||||
|
|
||||||
|
// 예: 2초 이내로 뒤로가기를 한 번 더 누르면 종료
|
||||||
|
static const _exitDuration = Duration(seconds: 2);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_isBackButtonVisible = false;
|
|
||||||
|
// (A) 메인페이지 들어올 때 전체 FRD 연결 해제
|
||||||
|
FirebaseDatabase.instance.goOffline();
|
||||||
|
|
||||||
|
// (B) 강제 종료 여부 확인 후 재입장 시도
|
||||||
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
|
_checkForcedExitStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
// (C) 배너 광고 초기화
|
||||||
|
_initBannerAd();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 배너 광고 초기화
|
||||||
|
void _initBannerAd() {
|
||||||
|
_bannerAd = BannerAd(
|
||||||
|
size: AdSize.banner, // 일반 배너 사이즈
|
||||||
|
// adUnitId: 'ca-app-pub-3940256099942544/6300978111' (테스트용)
|
||||||
|
adUnitId: 'ca-app-pub-3940256099942544/6300978111', // 예시: 구글 테스트 배너ID
|
||||||
|
listener: BannerAdListener(
|
||||||
|
onAdLoaded: (Ad ad) {
|
||||||
|
setState(() => _isBannerReady = true);
|
||||||
|
debugPrint('배너 광고 로드 완료');
|
||||||
|
},
|
||||||
|
onAdFailedToLoad: (Ad ad, LoadAdError err) {
|
||||||
|
debugPrint('배너 광고 로드 실패: $err');
|
||||||
|
ad.dispose();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
request: const AdRequest(),
|
||||||
|
);
|
||||||
|
|
||||||
|
// load() 호출로 광고 요청
|
||||||
|
_bannerAd?.load();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_bannerAd?.dispose(); // ★ 광고 자원 해제
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<bool> _onWillPop() async {
|
||||||
|
final now = DateTime.now();
|
||||||
|
if (_lastPressedTime == null ||
|
||||||
|
now.difference(_lastPressedTime!) > _exitDuration) {
|
||||||
|
// 첫 번째 뒤로가기 누름 or 이전 누름이 오래 전
|
||||||
|
_lastPressedTime = now;
|
||||||
|
|
||||||
|
// 안내 문구 띄우기 (Toast 예시)
|
||||||
|
Fluttertoast.showToast(
|
||||||
|
msg: '한 번 더 누르면 앱이 종료됩니다.',
|
||||||
|
toastLength: Toast.LENGTH_SHORT,
|
||||||
|
gravity: ToastGravity.BOTTOM,
|
||||||
|
);
|
||||||
|
return false; // 페이지 pop 하지 않음
|
||||||
|
}
|
||||||
|
// 2초 이내에 뒤로가기 두 번째 누름 → 앱 종료 허용
|
||||||
|
return true; // pop 허용 (Scaffold 밖으로 벗어남, 결과적으로 앱 종료)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// (B) 서버에 "강제 종료 여부" 확인 → 방이 있으면 재입장
|
||||||
|
Future<void> _checkForcedExitStatus() async {
|
||||||
|
try {
|
||||||
|
final Map<String, dynamic> requestBody = {};
|
||||||
|
final response = await Api.serverRequest(
|
||||||
|
uri: '/room/score/enter/running/room',
|
||||||
|
body: requestBody,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response['result'] == 'OK') {
|
||||||
|
final resp = response['response'] ?? {};
|
||||||
|
if (resp['result'] == 'OK') {
|
||||||
|
final data = resp['data'] ?? {};
|
||||||
|
final forceExitYn = (data['force_exit_yn'] ?? 'N').toString().toUpperCase();
|
||||||
|
if (forceExitYn == 'Y') {
|
||||||
|
final int roomSeq = data['room_seq'] ?? 0;
|
||||||
|
final String roomType = (data['room_type_name'] ?? '').toString().toUpperCase();
|
||||||
|
final String roomTitle = (data['room_title'] ?? '').toString();
|
||||||
|
|
||||||
|
// (1) 재입장 전 FRD 연결 복원
|
||||||
|
FirebaseDatabase.instance.goOnline();
|
||||||
|
|
||||||
|
showResponseDialog(context, '게임 재입장', '강제 종료 된 게임에 재입장 합니다.');
|
||||||
|
|
||||||
|
// (2) 방 타입에 따라 pushReplacement
|
||||||
|
if (roomType == 'TEAM') {
|
||||||
|
Navigator.pushReplacement(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (_) => PlayingTeamPage(
|
||||||
|
roomSeq: roomSeq,
|
||||||
|
roomTitle: roomTitle.isNotEmpty ? roomTitle : '재입장 (팀전)',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// 개인전
|
||||||
|
Navigator.pushReplacement(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (_) => PlayingPrivatePage(
|
||||||
|
roomSeq: roomSeq,
|
||||||
|
roomTitle: roomTitle.isNotEmpty ? roomTitle : '재입장 (개인전)',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
final msgTitle = resp['response_info']?['msg_title'] ?? '오류';
|
||||||
|
final msgContent = resp['response_info']?['msg_content'] ?? '강제 종료 여부 확인 실패';
|
||||||
|
showResponseDialog(context, msgTitle, msgContent);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showResponseDialog(context, '서버 오류', '강제 종료 여부 확인에 실패했습니다.');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
showResponseDialog(context, '서버 오류', '강제 종료 여부 확인에 실패했습니다.');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return WillPopScope(
|
||||||
// (A) 전체 배경 흰색 → 텍스트/버튼은 블랙 위주
|
onWillPop: _onWillPop,
|
||||||
|
child: Scaffold(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
|
|
||||||
// (B) 상단 AppBar: 블랙 배경, 흰색 아이콘
|
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
automaticallyImplyLeading: false, // 뒤로가기 버튼 자동생성 비활성
|
automaticallyImplyLeading: false, // 뒤로가기 버튼 X
|
||||||
title: const Text(
|
title: const Text(
|
||||||
'ALLSCORE',
|
'ALLSCORE',
|
||||||
style: TextStyle(color: Colors.white),
|
style: TextStyle(color: Colors.white),
|
||||||
@ -42,17 +180,16 @@ class _MainPageState extends State<MainPage> {
|
|||||||
IconButton(
|
IconButton(
|
||||||
icon: const Icon(Icons.settings, color: Colors.white),
|
icon: const Icon(Icons.settings, color: Colors.white),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
showSettingsDialog(context); // 설정 모달 호출
|
showSettingsDialog(context);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|
||||||
// (C) 본문: 위아래 공간 분배
|
|
||||||
body: Column(
|
body: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
// 중간 영역(“방 만들기” / “참여하기”)
|
// 중앙 버튼
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.all(16.0),
|
padding: const EdgeInsets.all(16.0),
|
||||||
@ -60,7 +197,6 @@ class _MainPageState extends State<MainPage> {
|
|||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
// (C1) 방 만들기 버튼
|
|
||||||
_buildBlackWhiteButton(
|
_buildBlackWhiteButton(
|
||||||
label: '방만들기',
|
label: '방만들기',
|
||||||
onTap: () {
|
onTap: () {
|
||||||
@ -71,10 +207,9 @@ class _MainPageState extends State<MainPage> {
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
const SizedBox(width: 16),
|
const SizedBox(width: 16),
|
||||||
// (C2) 참여하기 버튼 => RoomSearchHomePage로 이동
|
|
||||||
_buildBlackWhiteButton(
|
_buildBlackWhiteButton(
|
||||||
label: '참여하기',
|
label: '참여하기',
|
||||||
onTap: () async {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(builder: (_) => const RoomSearchHomePage()),
|
MaterialPageRoute(builder: (_) => const RoomSearchHomePage()),
|
||||||
@ -87,42 +222,38 @@ class _MainPageState extends State<MainPage> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
// (D) 광고 영역
|
// 광고 영역 교체
|
||||||
|
// (기존) Container(...) 대신 _bannerAd 위젯 사용
|
||||||
|
if (_isBannerReady && _bannerAd != null)
|
||||||
Container(
|
Container(
|
||||||
color: Colors.white,
|
width: _bannerAd!.size.width.toDouble(),
|
||||||
padding: const EdgeInsets.only(bottom: 24),
|
height: _bannerAd!.size.height.toDouble(),
|
||||||
child: Column(
|
alignment: Alignment.center,
|
||||||
mainAxisSize: MainAxisSize.min,
|
child: AdWidget(ad: _bannerAd!),
|
||||||
children: [
|
)
|
||||||
|
else
|
||||||
|
// 로딩중이거나 오류시 대체영역
|
||||||
Container(
|
Container(
|
||||||
height: 50,
|
|
||||||
width: 300,
|
width: 300,
|
||||||
|
height: 50,
|
||||||
color: Colors.grey.shade400,
|
color: Colors.grey.shade400,
|
||||||
child: const Center(
|
alignment: Alignment.center,
|
||||||
child: Text(
|
child: const Text(
|
||||||
'구글 광고',
|
'광고 로딩중',
|
||||||
style: TextStyle(color: Colors.black),
|
style: TextStyle(color: Colors.black),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
// (E) 임시 버튼: 방 생성 완료 이동
|
// 디버그용 임시버튼
|
||||||
Center(
|
Center(
|
||||||
child: OutlinedButton(
|
child: OutlinedButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
// 예시로 팀전 대기방(15번 방) 이동
|
// TODO
|
||||||
// 실무에서는 제외하거나 debugging용
|
|
||||||
// (아직 남겨두고 싶다면 유지)
|
|
||||||
},
|
},
|
||||||
style: OutlinedButton.styleFrom(
|
style: OutlinedButton.styleFrom(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
side: const BorderSide(color: Colors.black54, width: 1),
|
side: const BorderSide(color: Colors.black54, width: 1),
|
||||||
shape: RoundedRectangleBorder(
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(40)),
|
||||||
borderRadius: BorderRadius.circular(40),
|
|
||||||
),
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 60),
|
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 60),
|
||||||
),
|
),
|
||||||
child: const Text(
|
child: const Text(
|
||||||
@ -134,10 +265,10 @@ class _MainPageState extends State<MainPage> {
|
|||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 블랙 라인 + 흰 배경 스타일의 버튼
|
|
||||||
Widget _buildBlackWhiteButton({
|
Widget _buildBlackWhiteButton({
|
||||||
required String label,
|
required String label,
|
||||||
required VoidCallback onTap,
|
required VoidCallback onTap,
|
||||||
@ -149,9 +280,7 @@ class _MainPageState extends State<MainPage> {
|
|||||||
foregroundColor: Colors.black,
|
foregroundColor: Colors.black,
|
||||||
side: const BorderSide(color: Colors.black, width: 1),
|
side: const BorderSide(color: Colors.black, width: 1),
|
||||||
padding: const EdgeInsets.symmetric(vertical: 36, horizontal: 32),
|
padding: const EdgeInsets.symmetric(vertical: 36, horizontal: 32),
|
||||||
shape: RoundedRectangleBorder(
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
|
||||||
borderRadius: BorderRadius.circular(20),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
child: Text(label, style: const TextStyle(color: Colors.black)),
|
child: Text(label, style: const TextStyle(color: Colors.black)),
|
||||||
);
|
);
|
||||||
|
@ -3,14 +3,11 @@ import 'package:firebase_database/firebase_database.dart';
|
|||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
import 'main_page.dart';
|
import 'main_page.dart';
|
||||||
import 'finish_private_page.dart'; // (★) 개인전 종료화면
|
import 'finish_private_page.dart'; // 게임 종료 후 이동할 페이지
|
||||||
import '../../plugins/api.dart';
|
import '../../plugins/api.dart';
|
||||||
import '../../dialogs/response_dialog.dart';
|
import '../../dialogs/response_dialog.dart';
|
||||||
|
import '../../dialogs/score_edit_dialog.dart'; // 점수 수정 모달
|
||||||
// 점수 수정 모달
|
import '../../dialogs/user_info_basic_dialog.dart'; // 일반 유저 정보 모달
|
||||||
import '../../dialogs/score_edit_dialog.dart';
|
|
||||||
// 기존 사용자 정보 모달 (관리자/방장X)
|
|
||||||
import '../../dialogs/user_info_basic_dialog.dart';
|
|
||||||
|
|
||||||
class PlayingPrivatePage extends StatefulWidget {
|
class PlayingPrivatePage extends StatefulWidget {
|
||||||
final int roomSeq;
|
final int roomSeq;
|
||||||
@ -27,7 +24,6 @@ class PlayingPrivatePage extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
||||||
// FRD
|
|
||||||
late DatabaseReference _roomRef;
|
late DatabaseReference _roomRef;
|
||||||
Stream<DatabaseEvent>? _roomStream;
|
Stream<DatabaseEvent>? _roomStream;
|
||||||
|
|
||||||
@ -35,21 +31,20 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
String roomTitle = '';
|
String roomTitle = '';
|
||||||
|
|
||||||
int myScore = 0;
|
int myScore = 0;
|
||||||
|
|
||||||
// (ADMIN 제외) 플레이어 목록
|
|
||||||
List<Map<String, dynamic>> _scoreList = [];
|
List<Map<String, dynamic>> _scoreList = [];
|
||||||
|
|
||||||
bool _isLoading = true;
|
bool _isLoading = true;
|
||||||
|
|
||||||
// 내 user_seq
|
|
||||||
String mySeq = '0';
|
String mySeq = '0';
|
||||||
|
|
||||||
// userListMap
|
// userListMap: { userSeq: true/false }
|
||||||
Map<String, bool> _userListMap = {};
|
Map<String, bool> _userListMap = {};
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
// (1) FRD 연결 복원
|
||||||
|
FirebaseDatabase.instance.goOnline();
|
||||||
|
|
||||||
roomTitle = widget.roomTitle;
|
roomTitle = widget.roomTitle;
|
||||||
_initFirebase();
|
_initFirebase();
|
||||||
}
|
}
|
||||||
@ -79,18 +74,14 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final data = snapshot.value as Map<dynamic, dynamic>? ?? {};
|
final data = snapshot.value as Map<dynamic, dynamic>? ?? {};
|
||||||
|
|
||||||
final roomInfoData = data['roomInfo'] as Map<dynamic, dynamic>? ?? {};
|
final roomInfoData = data['roomInfo'] as Map<dynamic, dynamic>? ?? {};
|
||||||
final userInfoData = data['userInfo'] as Map<dynamic, dynamic>? ?? {};
|
final userInfoData = data['userInfo'] as Map<dynamic, dynamic>? ?? {};
|
||||||
final userListData = data['userList'] as Map<dynamic, dynamic>?;
|
final userListData = data['userList'] as Map<dynamic, dynamic>?;
|
||||||
|
|
||||||
// 방 상태 체크
|
// 방 상태 체크
|
||||||
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
||||||
|
|
||||||
// 만약 FINISH라면 => 종료 페이지 이동
|
|
||||||
if (roomStatus == 'FINISH') {
|
if (roomStatus == 'FINISH') {
|
||||||
// 모든 유저 -> 종료 페이지
|
// 종료 페이지
|
||||||
// (중복 이동 방지)
|
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
context,
|
context,
|
||||||
@ -117,7 +108,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 전체 참가자
|
// 전체 유저 목록
|
||||||
final List<Map<String, dynamic>> rawList = [];
|
final List<Map<String, dynamic>> rawList = [];
|
||||||
userInfoData.forEach((uSeq, uData) {
|
userInfoData.forEach((uSeq, uData) {
|
||||||
rawList.add({
|
rawList.add({
|
||||||
@ -133,10 +124,10 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// 내 점수
|
// 내 점수
|
||||||
int tempMyScore = 0;
|
int tmpMyScore = 0;
|
||||||
for (var u in rawList) {
|
for (var user in rawList) {
|
||||||
if ((u['is_my_score'] ?? 'N') == 'Y') {
|
if ((user['is_my_score'] ?? 'N') == 'Y') {
|
||||||
tempMyScore = u['score'] ?? 0;
|
tmpMyScore = user['score'] ?? 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -149,8 +140,9 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
return scoreB.compareTo(scoreA);
|
return scoreB.compareTo(scoreA);
|
||||||
});
|
});
|
||||||
|
|
||||||
myScore = tempMyScore;
|
myScore = tmpMyScore;
|
||||||
_scoreList = playerList;
|
_scoreList = playerList;
|
||||||
|
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
});
|
});
|
||||||
}, onError: (err) {
|
}, onError: (err) {
|
||||||
@ -161,11 +153,104 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (A) WillPopScope + AppBar leading
|
/// 방장이면 Finish API
|
||||||
Future<bool> _onBackPressed() async {
|
Future<void> _requestFinish() async {
|
||||||
// 방장? => 게임 종료 API
|
final reqBody = {
|
||||||
|
"room_seq": "${widget.roomSeq}",
|
||||||
|
"room_type": "PRIVATE",
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
await Api.serverRequest(uri: '/room/score/game/finish', body: reqBody);
|
||||||
|
} catch (e) {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 뒤로가기
|
||||||
|
Future<bool> _onWillPop() async {
|
||||||
if (roomMasterYn == 'Y') {
|
if (roomMasterYn == 'Y') {
|
||||||
|
// 방장 -> 모달
|
||||||
|
final confirm = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) {
|
||||||
|
return AlertDialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
side: const BorderSide(color: Colors.black, width: 1),
|
||||||
|
),
|
||||||
|
title: const Text('나가기', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||||
|
content: const Text(
|
||||||
|
'방장이 나가면 게임이 종료됩니다.\n정말 나가시겠습니까?',
|
||||||
|
style: TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, false),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('취소'),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, true),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('종료'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if (confirm != true) return false;
|
||||||
|
|
||||||
|
// Finish API
|
||||||
await _requestFinish();
|
await _requestFinish();
|
||||||
|
} else {
|
||||||
|
// 일반 유저
|
||||||
|
final confirm = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) {
|
||||||
|
return AlertDialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
side: const BorderSide(color: Colors.black, width: 1),
|
||||||
|
),
|
||||||
|
title: const Text('나가기', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||||
|
content: const Text(
|
||||||
|
'진행중인 방에서 나가면 재입장이 불가능합니다.\n정말 나가시겠습니까?',
|
||||||
|
style: TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, false),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('취소'),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, true),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('나가기'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if (confirm != true) return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// userList => false
|
// userList => false
|
||||||
@ -177,26 +262,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (B) 서버에 "게임 종료" 요청
|
/// 참가자 카드
|
||||||
Future<void> _requestFinish() async {
|
|
||||||
final reqBody = {
|
|
||||||
"room_seq": "${widget.roomSeq}",
|
|
||||||
"room_type": "PRIVATE",
|
|
||||||
};
|
|
||||||
try {
|
|
||||||
final resp = await Api.serverRequest(
|
|
||||||
uri: '/room/score/game/finish',
|
|
||||||
body: reqBody,
|
|
||||||
);
|
|
||||||
// OK / FAIL 등은 여기서 특별 처리 없이 넘어감
|
|
||||||
// room_status = FINISH => FRD에서 반영 -> 모든 참여자 이동
|
|
||||||
} catch (e) {
|
|
||||||
// 무시하거나 모달 표시
|
|
||||||
print('게임 종료 API 에러: $e');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// (C) 각 참가자 표시
|
|
||||||
Widget _buildScoreItem(Map<String, dynamic> user) {
|
Widget _buildScoreItem(Map<String, dynamic> user) {
|
||||||
final userSeq = user['user_seq'].toString();
|
final userSeq = user['user_seq'].toString();
|
||||||
final score = user['score'] ?? 0;
|
final score = user['score'] ?? 0;
|
||||||
@ -206,7 +272,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
final hasExited = !isActive;
|
final hasExited = !isActive;
|
||||||
|
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () => _onUserTapped(user),
|
onTap: () => _onTapUser(user),
|
||||||
child: Container(
|
child: Container(
|
||||||
width: 60,
|
width: 60,
|
||||||
margin: const EdgeInsets.all(4),
|
margin: const EdgeInsets.all(4),
|
||||||
@ -214,49 +280,43 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
hasExited
|
hasExited
|
||||||
? Text('X', style: TextStyle(fontSize: 20, color: Colors.redAccent, fontWeight: FontWeight.bold))
|
? Text('X', style: const TextStyle(fontSize: 20, color: Colors.redAccent, fontWeight: FontWeight.bold))
|
||||||
: Text('$score', style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold, color: Colors.black)),
|
: Text('$score', style: const TextStyle(fontSize: 20, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 2),
|
const SizedBox(height: 2),
|
||||||
Container(
|
Container(
|
||||||
width: 30,
|
width: 30, height: 30,
|
||||||
height: 30,
|
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
shape: BoxShape.circle,
|
shape: BoxShape.circle,
|
||||||
border: Border.all(color: hasExited ? Colors.redAccent : Colors.black),
|
border: Border.all(color: hasExited ? Colors.redAccent : Colors.black),
|
||||||
),
|
),
|
||||||
child: hasExited
|
child: hasExited
|
||||||
? Center(
|
? const Center(
|
||||||
child: Text('X', style: TextStyle(fontSize: 14, color: Colors.redAccent, fontWeight: FontWeight.bold)),
|
child: Text('X', style: TextStyle(fontSize: 14, color: Colors.redAccent, fontWeight: FontWeight.bold)),
|
||||||
)
|
)
|
||||||
: ClipOval(
|
: ClipOval(
|
||||||
child: Image.network(
|
child: Image.network(
|
||||||
'https://eldsoft.com:8097/images${user['profile_img']}',
|
'https://eldsoft.com:8097/images${user['profile_img']}',
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
errorBuilder: (ctx, err, st) => const Center(
|
errorBuilder: (_, __, ___) => const Center(child: Text('ERR', style: TextStyle(fontSize: 8))),
|
||||||
child: Text('ERR', style: TextStyle(fontSize: 8, color: Colors.black)),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 2),
|
const SizedBox(height: 2),
|
||||||
Text(
|
Text(nickname,
|
||||||
nickname,
|
|
||||||
style: TextStyle(fontSize: 11, color: hasExited ? Colors.redAccent : Colors.black),
|
style: TextStyle(fontSize: 11, color: hasExited ? Colors.redAccent : Colors.black),
|
||||||
overflow: TextOverflow.ellipsis,
|
overflow: TextOverflow.ellipsis),
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _onUserTapped(Map<String, dynamic> userData) async {
|
Future<void> _onTapUser(Map<String, dynamic> userData) async {
|
||||||
final pType = (userData['participant_type'] ?? '').toString().toUpperCase();
|
final pType = (userData['participant_type'] ?? '').toString().toUpperCase();
|
||||||
if (pType == 'ADMIN') {
|
if (pType == 'ADMIN') {
|
||||||
// 점수 수정 모달
|
// 점수수정
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => ScoreEditDialog(
|
builder: (_) => ScoreEditDialog(
|
||||||
roomSeq: widget.roomSeq,
|
roomSeq: widget.roomSeq,
|
||||||
roomType: 'PRIVATE',
|
roomType: 'PRIVATE',
|
||||||
@ -264,10 +324,9 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else if (roomMasterYn == 'Y') {
|
} else if (roomMasterYn == 'Y') {
|
||||||
// 방장(PLAYER)도 점수 수정
|
// 방장(PLAYER)도 수정 가능
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => ScoreEditDialog(
|
builder: (_) => ScoreEditDialog(
|
||||||
roomSeq: widget.roomSeq,
|
roomSeq: widget.roomSeq,
|
||||||
roomType: 'PRIVATE',
|
roomType: 'PRIVATE',
|
||||||
@ -275,10 +334,9 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
// 일반 모달
|
// 일반 유저 정보
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => UserInfoBasicDialog(userData: userData),
|
builder: (_) => UserInfoBasicDialog(userData: userData),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -287,7 +345,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return WillPopScope(
|
return WillPopScope(
|
||||||
onWillPop: _onBackPressed,
|
onWillPop: _onWillPop,
|
||||||
child: Scaffold(
|
child: Scaffold(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
@ -295,7 +353,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
elevation: 0,
|
elevation: 0,
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
||||||
onPressed: _onBackPressed,
|
onPressed: () => _onWillPop(),
|
||||||
),
|
),
|
||||||
title: Text(
|
title: Text(
|
||||||
roomTitle.isNotEmpty ? roomTitle : '진행중 (개인전)',
|
roomTitle.isNotEmpty ? roomTitle : '진행중 (개인전)',
|
||||||
@ -304,10 +362,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
actions: [
|
actions: [
|
||||||
if (roomMasterYn == 'Y')
|
if (roomMasterYn == 'Y')
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () async {
|
onPressed: _requestFinish,
|
||||||
// 방장 수동 종료버튼
|
|
||||||
await _requestFinish();
|
|
||||||
},
|
|
||||||
child: const Text('게임종료', style: TextStyle(color: Colors.white)),
|
child: const Text('게임종료', style: TextStyle(color: Colors.white)),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@ -319,22 +374,19 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
// 내 점수
|
// 내 점수
|
||||||
Container(
|
Container(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
color: Colors.white,
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 16),
|
padding: const EdgeInsets.symmetric(vertical: 16),
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
const Text('내 점수', style: TextStyle(fontSize: 14, fontWeight: FontWeight.bold, color: Colors.black)),
|
const Text('내 점수', style: TextStyle(fontSize: 14, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 4),
|
const SizedBox(height: 4),
|
||||||
Text('$myScore', style: const TextStyle(fontSize: 36, fontWeight: FontWeight.bold, color: Colors.black)),
|
Text('$myScore', style: const TextStyle(fontSize: 36, fontWeight: FontWeight.bold)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const Divider(height: 1, color: Colors.black),
|
const Divider(height: 1, color: Colors.black),
|
||||||
|
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.all(8),
|
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.all(8),
|
||||||
child: Wrap(
|
child: Wrap(
|
||||||
spacing: 8,
|
spacing: 8,
|
||||||
runSpacing: 8,
|
runSpacing: 8,
|
||||||
@ -342,8 +394,6 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
|
||||||
|
|
||||||
Container(
|
Container(
|
||||||
height: 50,
|
height: 50,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
@ -351,7 +401,7 @@ class _PlayingPrivatePageState extends State<PlayingPrivatePage> {
|
|||||||
border: Border.all(color: Colors.black, width: 1),
|
border: Border.all(color: Colors.black, width: 1),
|
||||||
),
|
),
|
||||||
child: const Center(
|
child: const Center(
|
||||||
child: Text('구글 광고', style: TextStyle(color: Colors.black)),
|
child: Text('구글 광고'),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
@ -3,7 +3,7 @@ import 'package:firebase_database/firebase_database.dart';
|
|||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
import 'main_page.dart';
|
import 'main_page.dart';
|
||||||
import 'finish_team_page.dart'; // (★) 팀전 종료화면
|
import 'finish_team_page.dart'; // 팀전 종료화면
|
||||||
import '../../plugins/api.dart';
|
import '../../plugins/api.dart';
|
||||||
import '../../dialogs/response_dialog.dart';
|
import '../../dialogs/response_dialog.dart';
|
||||||
import '../../dialogs/score_edit_dialog.dart';
|
import '../../dialogs/score_edit_dialog.dart';
|
||||||
@ -37,15 +37,17 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
Map<String, List<Map<String, dynamic>>> _teamMap = {};
|
Map<String, List<Map<String, dynamic>>> _teamMap = {};
|
||||||
|
|
||||||
bool _isLoading = true;
|
bool _isLoading = true;
|
||||||
|
|
||||||
String mySeq = '0';
|
String mySeq = '0';
|
||||||
|
|
||||||
// userListMap
|
// userListMap: { seq: true/false }
|
||||||
Map<String, bool> _userListMap = {};
|
Map<String, bool> _userListMap = {};
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
// (1) FRD 연결 복원
|
||||||
|
FirebaseDatabase.instance.goOnline();
|
||||||
|
|
||||||
roomTitle = widget.roomTitle;
|
roomTitle = widget.roomTitle;
|
||||||
_initFirebase();
|
_initFirebase();
|
||||||
}
|
}
|
||||||
@ -77,16 +79,13 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final data = snapshot.value as Map<dynamic, dynamic>? ?? {};
|
final data = snapshot.value as Map<dynamic, dynamic>? ?? {};
|
||||||
|
|
||||||
final roomInfoData = data['roomInfo'] as Map<dynamic, dynamic>? ?? {};
|
final roomInfoData = data['roomInfo'] as Map<dynamic, dynamic>? ?? {};
|
||||||
final userInfoData = data['userInfo'] as Map<dynamic, dynamic>? ?? {};
|
final userInfoData = data['userInfo'] as Map<dynamic, dynamic>? ?? {};
|
||||||
final userListData = data['userList'] as Map<dynamic, dynamic>?;
|
final userListData = data['userList'] as Map<dynamic, dynamic>?;
|
||||||
|
|
||||||
// room_status
|
|
||||||
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
||||||
|
|
||||||
// FINISH -> 종료화면
|
|
||||||
if (roomStatus == 'FINISH') {
|
if (roomStatus == 'FINISH') {
|
||||||
|
// 종료화면
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
context,
|
context,
|
||||||
@ -97,14 +96,12 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
setState(() {
|
setState(() {
|
||||||
// 방장 여부
|
|
||||||
final masterSeq = roomInfoData['master_user_seq'];
|
final masterSeq = roomInfoData['master_user_seq'];
|
||||||
roomMasterYn = (masterSeq != null && masterSeq.toString() == mySeq) ? 'Y' : 'N';
|
roomMasterYn = (masterSeq != null && masterSeq.toString() == mySeq) ? 'Y' : 'N';
|
||||||
|
|
||||||
final newTitle = (roomInfoData['room_title'] ?? '') as String;
|
final newTitle = (roomInfoData['room_title'] ?? '') as String;
|
||||||
if (newTitle.isNotEmpty) roomTitle = newTitle;
|
if (newTitle.isNotEmpty) roomTitle = newTitle;
|
||||||
|
|
||||||
// userListMap
|
|
||||||
_userListMap.clear();
|
_userListMap.clear();
|
||||||
if (userListData != null) {
|
if (userListData != null) {
|
||||||
userListData.forEach((k, v) {
|
userListData.forEach((k, v) {
|
||||||
@ -124,7 +121,7 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// 내 점수/팀점수
|
// 내 점수 & 팀 점수
|
||||||
int tmpMyScore = 0;
|
int tmpMyScore = 0;
|
||||||
int tmpMyTeamScore = 0;
|
int tmpMyTeamScore = 0;
|
||||||
String myTeam = 'WAIT';
|
String myTeam = 'WAIT';
|
||||||
@ -132,29 +129,28 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
for (var user in rawList) {
|
for (var user in rawList) {
|
||||||
final uSeq = user['user_seq'].toString();
|
final uSeq = user['user_seq'].toString();
|
||||||
final sc = (user['score'] ?? 0) as int;
|
final sc = (user['score'] ?? 0) as int;
|
||||||
final tName = user['team_name'] ?? 'WAIT';
|
final tName= (user['team_name'] ?? 'WAIT');
|
||||||
if (uSeq == mySeq) {
|
if (uSeq == mySeq) {
|
||||||
tmpMyScore = sc;
|
tmpMyScore = sc;
|
||||||
myTeam = tName;
|
myTeam = tName;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 내 팀 점수
|
|
||||||
for (var user in rawList) {
|
for (var user in rawList) {
|
||||||
final tName = user['team_name'] ?? 'WAIT';
|
final tName = user['team_name'] ?? 'WAIT';
|
||||||
final sc = (user['score'] ?? 0) as int;
|
final sc = (user['score'] ?? 0) as int;
|
||||||
if (tName == myTeam && tName != 'WAIT') {
|
if (tName == myTeam && myTeam != 'WAIT') {
|
||||||
tmpMyTeamScore += sc;
|
tmpMyTeamScore += sc;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 팀별 분류 (ADMIN/WAIT 제외)
|
// ADMIN, WAIT 제외
|
||||||
final Map<String, List<Map<String, dynamic>>> tMap = {};
|
final Map<String, List<Map<String, dynamic>>> tMap = {};
|
||||||
final Map<String, int> tScoreMap = {};
|
final Map<String, int> tScoreMap = {};
|
||||||
|
|
||||||
for (var user in rawList) {
|
for (var user in rawList) {
|
||||||
final pType = user['participant_type'];
|
final pType = user['participant_type'];
|
||||||
final tName = user['team_name'] ?? 'WAIT';
|
final tName = (user['team_name'] ?? 'WAIT');
|
||||||
if (pType == 'ADMIN') continue;
|
if (pType == 'ADMIN') continue;
|
||||||
if (tName == 'WAIT') continue;
|
if (tName == 'WAIT') continue;
|
||||||
|
|
||||||
@ -162,7 +158,6 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
tMap[tName]!.add(user);
|
tMap[tName]!.add(user);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 팀 점수 합
|
|
||||||
tMap.forEach((k, members) {
|
tMap.forEach((k, members) {
|
||||||
int sumScore = 0;
|
int sumScore = 0;
|
||||||
for (var m in members) {
|
for (var m in members) {
|
||||||
@ -175,6 +170,7 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
myTeamScore = tmpMyTeamScore;
|
myTeamScore = tmpMyTeamScore;
|
||||||
_teamMap = tMap;
|
_teamMap = tMap;
|
||||||
_teamScoreMap = tScoreMap;
|
_teamScoreMap = tScoreMap;
|
||||||
|
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
});
|
});
|
||||||
}, onError: (err) {
|
}, onError: (err) {
|
||||||
@ -185,11 +181,102 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (A) 뒤로가기 -> 방장? => Finish API
|
/// 게임종료
|
||||||
Future<bool> _onBackPressed() async {
|
Future<void> _requestFinish() async {
|
||||||
if (roomMasterYn == 'Y') {
|
final body = {
|
||||||
await _requestFinish();
|
"room_seq": "${widget.roomSeq}",
|
||||||
|
"room_type": "TEAM",
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
await Api.serverRequest(uri: '/room/score/game/finish', body: body);
|
||||||
|
} catch (e) {
|
||||||
|
// ignore
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<bool> _onWillPop() async {
|
||||||
|
if (roomMasterYn == 'Y') {
|
||||||
|
// 방장 모달
|
||||||
|
final confirm = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) {
|
||||||
|
return AlertDialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
side: const BorderSide(color: Colors.black, width: 1),
|
||||||
|
),
|
||||||
|
title: const Text('나가기', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||||
|
content: const Text(
|
||||||
|
'방장이 나가면 게임이 종료됩니다.\n정말 나가시겠습니까?',
|
||||||
|
style: TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, false),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('취소'),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, true),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('종료'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (confirm != true) return false;
|
||||||
|
|
||||||
|
await _requestFinish();
|
||||||
|
} else {
|
||||||
|
// 일반 유저
|
||||||
|
final confirm = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (_) {
|
||||||
|
return AlertDialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
side: const BorderSide(color: Colors.black, width: 1),
|
||||||
|
),
|
||||||
|
title: const Text('나가기', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||||
|
content: const Text(
|
||||||
|
'진행중인 방에서 나가면 재입장이 불가능합니다.\n정말 나가시겠습니까?',
|
||||||
|
style: TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, false),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('취소'),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, true),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
),
|
||||||
|
child: const Text('나가기'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (confirm != true) return false;
|
||||||
|
}
|
||||||
|
|
||||||
// userList => false
|
// userList => false
|
||||||
final userRef = _roomRef.child('userList').child(mySeq);
|
final userRef = _roomRef.child('userList').child(mySeq);
|
||||||
await userRef.set(false);
|
await userRef.set(false);
|
||||||
@ -199,26 +286,10 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _requestFinish() async {
|
|
||||||
final body = {
|
|
||||||
"room_seq": "${widget.roomSeq}",
|
|
||||||
"room_type": "TEAM",
|
|
||||||
};
|
|
||||||
try {
|
|
||||||
final resp = await Api.serverRequest(
|
|
||||||
uri: '/room/score/game/finish',
|
|
||||||
body: body,
|
|
||||||
);
|
|
||||||
// result ...
|
|
||||||
} catch (e) {
|
|
||||||
print('finish API error: $e');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return WillPopScope(
|
return WillPopScope(
|
||||||
onWillPop: _onBackPressed,
|
onWillPop: _onWillPop,
|
||||||
child: Scaffold(
|
child: Scaffold(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
@ -230,7 +301,7 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
elevation: 0,
|
elevation: 0,
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
||||||
onPressed: _onBackPressed,
|
onPressed: () => _onWillPop(),
|
||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
if (roomMasterYn == 'Y')
|
if (roomMasterYn == 'Y')
|
||||||
@ -244,7 +315,7 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
? const Center(child: CircularProgressIndicator())
|
? const Center(child: CircularProgressIndicator())
|
||||||
: Column(
|
: Column(
|
||||||
children: [
|
children: [
|
||||||
// 내 점수 / 팀 점수
|
// (A) 내 점수 / 팀 점수
|
||||||
Container(
|
Container(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
padding: const EdgeInsets.only(top: 16, bottom: 16),
|
padding: const EdgeInsets.only(top: 16, bottom: 16),
|
||||||
@ -253,17 +324,17 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
children: [
|
children: [
|
||||||
Column(
|
Column(
|
||||||
children: [
|
children: [
|
||||||
const Text('내 점수', style: TextStyle(fontSize: 14, fontWeight: FontWeight.bold, color: Colors.black)),
|
const Text('내 점수', style: TextStyle(fontSize: 14, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 4),
|
const SizedBox(height: 4),
|
||||||
Text('$myScore', style: const TextStyle(fontSize: 36, fontWeight: FontWeight.bold, color: Colors.black)),
|
Text('$myScore', style: const TextStyle(fontSize: 36, fontWeight: FontWeight.bold)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
Container(width: 1, height: 60, color: Colors.black),
|
Container(width: 1, height: 60, color: Colors.black),
|
||||||
Column(
|
Column(
|
||||||
children: [
|
children: [
|
||||||
const Text('우리 팀 점수', style: TextStyle(fontSize: 14, fontWeight: FontWeight.bold, color: Colors.black)),
|
const Text('우리 팀 점수', style: TextStyle(fontSize: 14, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 4),
|
const SizedBox(height: 4),
|
||||||
Text('$myTeamScore', style: const TextStyle(fontSize: 36, fontWeight: FontWeight.bold, color: Colors.black)),
|
Text('$myTeamScore', style: const TextStyle(fontSize: 36, fontWeight: FontWeight.bold)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@ -271,6 +342,7 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
),
|
),
|
||||||
const Divider(height: 1, color: Colors.black),
|
const Divider(height: 1, color: Colors.black),
|
||||||
|
|
||||||
|
// (B) 팀별 표시
|
||||||
Expanded(
|
Expanded(
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
padding: const EdgeInsets.all(8),
|
padding: const EdgeInsets.all(8),
|
||||||
@ -280,6 +352,7 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
Container(
|
Container(
|
||||||
height: 50,
|
height: 50,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
@ -316,7 +389,8 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
padding: const EdgeInsets.symmetric(vertical: 8),
|
padding: const EdgeInsets.symmetric(vertical: 8),
|
||||||
child: Center(
|
child: Center(
|
||||||
child: Text('$teamName (팀점수 $teamScore)', style: const TextStyle(color: Colors.white, fontWeight: FontWeight.bold)),
|
child: Text('$teamName (팀점수 $teamScore)',
|
||||||
|
style: const TextStyle(color: Colors.white, fontWeight: FontWeight.bold)),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Container(
|
Container(
|
||||||
@ -334,13 +408,13 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
Widget _buildTeamMemberItem(Map<String, dynamic> userData) {
|
Widget _buildTeamMemberItem(Map<String, dynamic> userData) {
|
||||||
final userSeq = userData['user_seq'].toString();
|
final userSeq = userData['user_seq'].toString();
|
||||||
final score = userData['score'] ?? 0;
|
final score = userData['score'] ?? 0;
|
||||||
final nickname = userData['nickname'] ?? '유저';
|
final nickname= userData['nickname'] ?? '유저';
|
||||||
|
|
||||||
final bool isActive = _userListMap[userSeq] ?? true;
|
final bool isActive = _userListMap[userSeq] ?? true;
|
||||||
final hasExited = !isActive;
|
final hasExited = !isActive;
|
||||||
|
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () => _onUserTapped(userData),
|
onTap: () => _onTapUser(userData),
|
||||||
child: Container(
|
child: Container(
|
||||||
width: 60,
|
width: 60,
|
||||||
margin: const EdgeInsets.symmetric(horizontal: 6, vertical: 4),
|
margin: const EdgeInsets.symmetric(horizontal: 6, vertical: 4),
|
||||||
@ -348,8 +422,8 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
hasExited
|
hasExited
|
||||||
? Text('X', style: TextStyle(fontSize: 18, color: Colors.redAccent, fontWeight: FontWeight.bold))
|
? Text('X', style: const TextStyle(fontSize: 18, color: Colors.redAccent, fontWeight: FontWeight.bold))
|
||||||
: Text('$score', style: const TextStyle(fontSize: 18, fontWeight: FontWeight.bold, color: Colors.black)),
|
: Text('$score', style: const TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 2),
|
const SizedBox(height: 2),
|
||||||
Container(
|
Container(
|
||||||
width: 30,
|
width: 30,
|
||||||
@ -359,12 +433,18 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
border: Border.all(color: hasExited ? Colors.redAccent : Colors.black),
|
border: Border.all(color: hasExited ? Colors.redAccent : Colors.black),
|
||||||
),
|
),
|
||||||
child: hasExited
|
child: hasExited
|
||||||
? Center(child: Text('X', style: TextStyle(fontSize: 14, color: Colors.redAccent, fontWeight: FontWeight.bold)))
|
? const Center(
|
||||||
|
child: Text(
|
||||||
|
'X',
|
||||||
|
style: TextStyle(fontSize: 14, color: Colors.redAccent, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
)
|
||||||
: ClipOval(
|
: ClipOval(
|
||||||
child: Image.network(
|
child: Image.network(
|
||||||
'https://eldsoft.com:8097/images${userData['profile_img']}',
|
'https://eldsoft.com:8097/images${userData['profile_img']}',
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
errorBuilder: (ctx, err, st) => const Center(child: Text('ERR', style: TextStyle(fontSize: 8, color: Colors.black))),
|
errorBuilder: (ctx, err, st) =>
|
||||||
|
const Center(child: Text('ERR', style: TextStyle(fontSize: 8))),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -380,13 +460,11 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _onUserTapped(Map<String, dynamic> userData) async {
|
Future<void> _onTapUser(Map<String, dynamic> userData) async {
|
||||||
final pType = (userData['participant_type'] ?? '').toString().toUpperCase();
|
final pType = (userData['participant_type'] ?? '').toString().toUpperCase();
|
||||||
if (pType == 'ADMIN') {
|
if (pType == 'ADMIN') {
|
||||||
// 점수 수정
|
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => ScoreEditDialog(
|
builder: (_) => ScoreEditDialog(
|
||||||
roomSeq: widget.roomSeq,
|
roomSeq: widget.roomSeq,
|
||||||
roomType: 'TEAM',
|
roomType: 'TEAM',
|
||||||
@ -396,7 +474,6 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
} else if (roomMasterYn == 'Y') {
|
} else if (roomMasterYn == 'Y') {
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => ScoreEditDialog(
|
builder: (_) => ScoreEditDialog(
|
||||||
roomSeq: widget.roomSeq,
|
roomSeq: widget.roomSeq,
|
||||||
roomType: 'TEAM',
|
roomType: 'TEAM',
|
||||||
@ -406,7 +483,6 @@ class _PlayingTeamPageState extends State<PlayingTeamPage> {
|
|||||||
} else {
|
} else {
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => UserInfoBasicDialog(userData: userData),
|
builder: (_) => UserInfoBasicDialog(userData: userData),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
import '../../plugins/api.dart'; // 서버 요청용 (예: Api.serverRequest)
|
import '../../plugins/api.dart';
|
||||||
import '../../dialogs/response_dialog.dart'; // 모달창 띄우기 예시
|
import '../../dialogs/response_dialog.dart';
|
||||||
import '../../dialogs/room_detail_dialog.dart'; // 분리된 모달창 import
|
import '../../dialogs/room_detail_dialog.dart';
|
||||||
|
|
||||||
|
// (새로 import)
|
||||||
|
// 종료된 방 페이지들
|
||||||
|
import '../room/finish_private_page.dart';
|
||||||
|
import '../room/finish_team_page.dart';
|
||||||
|
|
||||||
/// 서버로부터 방 리스트를 검색/조회하는 페이지
|
|
||||||
/// - roomStatus: "WAIT"/"RUNNING"/"FINISH"
|
|
||||||
/// - 1페이지당 10개씩 로드, 스크롤 최하단 도달 시 다음 페이지 자동 로드
|
|
||||||
/// - 검색창을 통해 room_title 필터링
|
|
||||||
class RoomSearchListPage extends StatefulWidget {
|
class RoomSearchListPage extends StatefulWidget {
|
||||||
final String roomStatus; // WAIT / RUNNING / FINISH
|
final String roomStatus; // WAIT / RUNNING / FINISH
|
||||||
|
|
||||||
@ -21,7 +22,6 @@ class RoomSearchListPage extends StatefulWidget {
|
|||||||
class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
||||||
final TextEditingController _searchController = TextEditingController();
|
final TextEditingController _searchController = TextEditingController();
|
||||||
|
|
||||||
// 방 목록
|
|
||||||
List<Map<String, dynamic>> _roomList = [];
|
List<Map<String, dynamic>> _roomList = [];
|
||||||
|
|
||||||
bool _isLoading = false;
|
bool _isLoading = false;
|
||||||
@ -47,7 +47,6 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 스크롤이 최하단 근처 도달 시 다음 페이지 로드
|
|
||||||
void _onScroll() {
|
void _onScroll() {
|
||||||
if (!_scrollController.hasClients) return;
|
if (!_scrollController.hasClients) return;
|
||||||
final thresholdPixels = 200;
|
final thresholdPixels = 200;
|
||||||
@ -59,7 +58,6 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (1) 서버에서 방 리스트 가져오기
|
|
||||||
Future<void> _fetchRoomList({required bool isRefresh}) async {
|
Future<void> _fetchRoomList({required bool isRefresh}) async {
|
||||||
if (_isLoading) return;
|
if (_isLoading) return;
|
||||||
if (!isRefresh && !_hasMore) return;
|
if (!isRefresh && !_hasMore) return;
|
||||||
@ -72,7 +70,6 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
_roomList.clear();
|
_roomList.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
// 서버 API 요구사항에 맞춰 WAIT/RUNNING/FINISH (대문자) 사용
|
|
||||||
final String searchType = widget.roomStatus.toUpperCase();
|
final String searchType = widget.roomStatus.toUpperCase();
|
||||||
final String searchValue = _searchController.text.trim();
|
final String searchValue = _searchController.text.trim();
|
||||||
final String searchPage = _currentPage.toString();
|
final String searchPage = _currentPage.toString();
|
||||||
@ -84,14 +81,8 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final response = await Api.serverRequest(
|
final response = await Api.serverRequest(uri: '/room/score/room/list', body: requestBody);
|
||||||
uri: '/room/score/room/list',
|
|
||||||
body: requestBody,
|
|
||||||
);
|
|
||||||
|
|
||||||
print('🔵 response: $response');
|
|
||||||
|
|
||||||
// (참고) 서버 구조: { result: OK, response: {...}, ... }
|
|
||||||
if (response == null || response['result'] != 'OK') {
|
if (response == null || response['result'] != 'OK') {
|
||||||
showResponseDialog(context, '오류', '방 목록을 불러오지 못했습니다.');
|
showResponseDialog(context, '오류', '방 목록을 불러오지 못했습니다.');
|
||||||
} else {
|
} else {
|
||||||
@ -105,13 +96,14 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
_hasMore = false;
|
_hasMore = false;
|
||||||
} else {
|
} else {
|
||||||
for (var item in respData) {
|
for (var item in respData) {
|
||||||
print('🔵 item: $item');
|
|
||||||
final parsedItem = {
|
final parsedItem = {
|
||||||
'room_seq': item['room_seq'] ?? 0,
|
'room_seq': item['room_seq'] ?? 0,
|
||||||
'nickname': item['nickname'] ?? '사용자',
|
'nickname': item['nickname'] ?? '사용자',
|
||||||
|
// WAIT/RUNNING/FINISH -> 한글
|
||||||
'room_status': _statusToKr(item['room_status'] ?? ''),
|
'room_status': _statusToKr(item['room_status'] ?? ''),
|
||||||
|
'raw_room_status': (item['room_status'] ?? '').toString().toUpperCase(),
|
||||||
'open_yn': (item['open_yn'] == 'Y') ? '공개' : '비공개',
|
'open_yn': (item['open_yn'] == 'Y') ? '공개' : '비공개',
|
||||||
'room_type': item['room_type_name'] ?? 'private',
|
'room_type': (item['room_type_name'] ?? 'PRIVATE').toString().toLowerCase(),
|
||||||
'room_title': item['room_title'] ?? '(방제목 없음)',
|
'room_title': item['room_title'] ?? '(방제목 없음)',
|
||||||
'room_intro': item['room_intro'] ?? '',
|
'room_intro': item['room_intro'] ?? '',
|
||||||
'now_people': item['now_number_of_people']?.toString() ?? '0',
|
'now_people': item['now_number_of_people']?.toString() ?? '0',
|
||||||
@ -137,7 +129,6 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// WAIT->'대기중', RUNNING->'진행중', FINISH->'종료'
|
|
||||||
String _statusToKr(String status) {
|
String _statusToKr(String status) {
|
||||||
switch (status.toUpperCase()) {
|
switch (status.toUpperCase()) {
|
||||||
case 'WAIT':
|
case 'WAIT':
|
||||||
@ -156,13 +147,43 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _onRoomItemTap(Map<String, dynamic> item) {
|
void _onRoomItemTap(Map<String, dynamic> item) {
|
||||||
// 여기서 분리된 모달 호출
|
// room_status(한글)이 아니라, raw_room_status(영문 WAIT/RUNNING/FINISH)를 보고 판단
|
||||||
|
final rawStatus = (item['raw_room_status'] ?? '').toString().toUpperCase();
|
||||||
|
if (rawStatus == 'FINISH') {
|
||||||
|
// 종료된 방이면 => FinishPrivatePage or FinishTeamPage
|
||||||
|
final roomType = (item['room_type'] ?? 'private').toString().toLowerCase();
|
||||||
|
final roomSeq = (item['room_seq'] ?? 0) as int;
|
||||||
|
final roomTitle = (item['room_title'] ?? '(종료된 방)') as String;
|
||||||
|
|
||||||
|
if (roomType == 'private') {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (_) => FinishPrivatePage(
|
||||||
|
roomSeq: roomSeq,
|
||||||
|
fromPlayingPage: false, // ← 검색에서 왔으므로 false
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (_) => FinishTeamPage(
|
||||||
|
roomSeq: roomSeq,
|
||||||
|
fromPlayingPage: false,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 아직 진행중 or 대기중인 방 => 기존 로직, 예: RoomDetailDialog
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (_) => RoomDetailDialog(roomData: item),
|
builder: (_) => RoomDetailDialog(roomData: item),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@ -180,7 +201,7 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
),
|
),
|
||||||
body: Column(
|
body: Column(
|
||||||
children: [
|
children: [
|
||||||
// (A) 검색창
|
// 검색창
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.all(8.0),
|
padding: const EdgeInsets.all(8.0),
|
||||||
child: TextField(
|
child: TextField(
|
||||||
@ -206,14 +227,12 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
// (B) 로딩 표시 or 리스트
|
|
||||||
Expanded(
|
Expanded(
|
||||||
child: _isLoading && _roomList.isEmpty
|
child: _isLoading && _roomList.isEmpty
|
||||||
? const Center(child: CircularProgressIndicator())
|
? const Center(child: CircularProgressIndicator())
|
||||||
: _buildRoomListView(),
|
: _buildRoomListView(),
|
||||||
),
|
),
|
||||||
|
|
||||||
// (C) 하단 광고
|
|
||||||
Container(
|
Container(
|
||||||
height: 60,
|
height: 60,
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
@ -234,16 +253,11 @@ class _RoomSearchListPageState extends State<RoomSearchListPage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildRoomListView() {
|
Widget _buildRoomListView() {
|
||||||
print('🔵 _roomList: $_roomList');
|
|
||||||
if (_roomList.isEmpty) {
|
if (_roomList.isEmpty) {
|
||||||
return const Center(
|
return const Center(
|
||||||
child: Text(
|
child: Text('검색 결과가 없습니다.', style: TextStyle(color: Colors.black)),
|
||||||
'검색 결과가 없습니다.',
|
|
||||||
style: TextStyle(color: Colors.black),
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return ListView.builder(
|
return ListView.builder(
|
||||||
controller: _scrollController,
|
controller: _scrollController,
|
||||||
padding: const EdgeInsets.symmetric(vertical: 8, horizontal: 16),
|
padding: const EdgeInsets.symmetric(vertical: 8, horizontal: 16),
|
||||||
|
@ -4,9 +4,9 @@ import 'package:firebase_database/firebase_database.dart';
|
|||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
import 'main_page.dart';
|
import 'main_page.dart';
|
||||||
import '../../plugins/api.dart'; // 서버 API
|
import '../../plugins/api.dart';
|
||||||
import '../../dialogs/response_dialog.dart';
|
import '../../dialogs/response_dialog.dart';
|
||||||
import '../../dialogs/yes_no_dialog.dart'; // 예/아니오 모달
|
import '../../dialogs/yes_no_dialog.dart';
|
||||||
import '../../dialogs/room_setting_dialog.dart';
|
import '../../dialogs/room_setting_dialog.dart';
|
||||||
import '../../dialogs/user_info_private_dialog.dart';
|
import '../../dialogs/user_info_private_dialog.dart';
|
||||||
import 'playing_private_page.dart';
|
import 'playing_private_page.dart';
|
||||||
@ -38,46 +38,57 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
int numberOfPeople = 10;
|
int numberOfPeople = 10;
|
||||||
String scoreOpenRange = 'PRIVATE';
|
String scoreOpenRange = 'PRIVATE';
|
||||||
|
|
||||||
|
// FRD
|
||||||
|
late DatabaseReference _roomRef;
|
||||||
|
Stream<DatabaseEvent>? _roomStream;
|
||||||
|
StreamSubscription<DatabaseEvent>? _roomStreamSubscription;
|
||||||
|
|
||||||
// 유저 목록
|
// 유저 목록
|
||||||
List<Map<String, dynamic>> _userList = [];
|
List<Map<String, dynamic>> _userList = [];
|
||||||
|
|
||||||
bool _isLoading = true;
|
bool _isLoading = true;
|
||||||
|
|
||||||
// FRD
|
|
||||||
late DatabaseReference _roomRef;
|
|
||||||
Stream<DatabaseEvent>? _roomStream;
|
|
||||||
|
|
||||||
// 진행중 화면 이동 중복 방지
|
// 진행중 화면 이동 중복 방지
|
||||||
bool _movedToRunningPage = false;
|
bool _movedToRunningPage = false;
|
||||||
|
|
||||||
// 강퇴 안내 중복 방지
|
// 강퇴 안내 중복 방지
|
||||||
bool _kickedOut = false;
|
bool _kickedOut = false;
|
||||||
|
|
||||||
// FRD 구독 해제
|
// 내 user_seq
|
||||||
StreamSubscription<DatabaseEvent>? _roomStreamSubscription;
|
String mySeq = '0';
|
||||||
|
|
||||||
// (예) 내 user_seq
|
// ─────────────────────────────────────────
|
||||||
String mySeq = '0'; // 원래 '6' 고정이었던 부분 제거
|
// 1시간 카운트다운
|
||||||
|
// ─────────────────────────────────────────
|
||||||
|
Timer? _countdownTimer;
|
||||||
|
Duration _remaining = const Duration(hours: 1); // 기본 1시간
|
||||||
|
DateTime? _createDt; // FRD의 roomInfo.create_dt
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_loadMySeq();
|
// FRD 연결 복원
|
||||||
|
FirebaseDatabase.instance.goOnline();
|
||||||
|
_initRoomRef();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (A) my_user_seq 로드 -> 리스너
|
Future<void> _initRoomRef() async {
|
||||||
Future<void> _loadMySeq() async {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
mySeq = prefs.getInt('my_user_seq')?.toString() ?? '0';
|
mySeq = prefs.getInt('my_user_seq')?.toString() ?? '0';
|
||||||
|
|
||||||
final roomKey = 'korea-${widget.roomSeq}';
|
final roomKey = 'korea-${widget.roomSeq}';
|
||||||
_roomRef = FirebaseDatabase.instance.ref('rooms/$roomKey');
|
_roomRef = FirebaseDatabase.instance.ref('rooms/$roomKey');
|
||||||
|
|
||||||
|
// onDisconnect + connect_yn='Y'
|
||||||
|
final myUserRef = _roomRef.child('userInfo').child(mySeq);
|
||||||
|
myUserRef.onDisconnect().update({'connect_yn': 'N'});
|
||||||
|
await myUserRef.update({'connect_yn': 'Y'});
|
||||||
|
|
||||||
_listenRoomData();
|
_listenRoomData();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _listenRoomData() {
|
void _listenRoomData() {
|
||||||
_roomStream = _roomRef.onValue;
|
_roomStream = _roomRef.onValue;
|
||||||
_roomStream?.listen((event) {
|
_roomStreamSubscription = _roomStream?.listen((event) {
|
||||||
final snapshot = event.snapshot;
|
final snapshot = event.snapshot;
|
||||||
if (!snapshot.exists) {
|
if (!snapshot.exists) {
|
||||||
setState(() {
|
setState(() {
|
||||||
@ -94,6 +105,7 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
|
|
||||||
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
||||||
|
|
||||||
|
// (A) roomInfo 갱신
|
||||||
setState(() {
|
setState(() {
|
||||||
roomTitle = (roomInfoData['room_title'] ?? '') as String;
|
roomTitle = (roomInfoData['room_title'] ?? '') as String;
|
||||||
roomIntro = (roomInfoData['room_intro'] ?? '') as String;
|
roomIntro = (roomInfoData['room_intro'] ?? '') as String;
|
||||||
@ -110,7 +122,7 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
roomMasterYn = 'Y';
|
roomMasterYn = 'Y';
|
||||||
}
|
}
|
||||||
|
|
||||||
// 유저 목록
|
// userList
|
||||||
final tempList = <Map<String, dynamic>>[];
|
final tempList = <Map<String, dynamic>>[];
|
||||||
userInfoData.forEach((userSeq, userMap) {
|
userInfoData.forEach((userSeq, userMap) {
|
||||||
tempList.add({
|
tempList.add({
|
||||||
@ -119,16 +131,16 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
'nickname': userMap['nickname'] ?? '유저',
|
'nickname': userMap['nickname'] ?? '유저',
|
||||||
'score': userMap['score'] ?? 0,
|
'score': userMap['score'] ?? 0,
|
||||||
'profile_img': userMap['profile_img'] ?? '',
|
'profile_img': userMap['profile_img'] ?? '',
|
||||||
'department': userMap['department'] ?? '',
|
|
||||||
'introduce_myself': userMap['introduce_myself'] ?? '',
|
'introduce_myself': userMap['introduce_myself'] ?? '',
|
||||||
'ready_yn': (userMap['ready_yn'] ?? 'N').toString().toUpperCase(),
|
'ready_yn': (userMap['ready_yn'] ?? 'N').toString().toUpperCase(),
|
||||||
|
'connect_yn': (userMap['connect_yn'] ?? 'Y').toString().toUpperCase(),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
_userList = tempList;
|
_userList = tempList;
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
});
|
});
|
||||||
|
|
||||||
// 진행중 -> 화면 이동
|
// (B) 진행중 => 이동
|
||||||
if (roomStatus == 'RUNNING' && !_movedToRunningPage) {
|
if (roomStatus == 'RUNNING' && !_movedToRunningPage) {
|
||||||
_movedToRunningPage = true;
|
_movedToRunningPage = true;
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
@ -143,20 +155,21 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// (2) 여기서 "내 user_seq가 목록에 있는지" 검사
|
// (C) 1시간 카운트다운 위한 create_dt 파싱
|
||||||
|
// 예: "2025-01-07T06:38:10.123456"
|
||||||
|
final createDtStr = (roomInfoData['create_dt'] ?? '') as String;
|
||||||
|
if (createDtStr.isNotEmpty && createDtStr.contains('T')) {
|
||||||
|
final dt = DateTime.tryParse(createDtStr);
|
||||||
|
if (dt != null) {
|
||||||
|
_createDt = dt;
|
||||||
|
_startCountdownTimer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// (D) 내가 목록에서 사라졌는지 => 강퇴 판별
|
||||||
final amIStillHere = _userList.any((u) => u['user_seq'].toString() == mySeq);
|
final amIStillHere = _userList.any((u) => u['user_seq'].toString() == mySeq);
|
||||||
|
|
||||||
// (3) 만약 내가 목록에서 사라졌고,
|
|
||||||
// 아직 안내하지 않았으며(_kickedOut == false),
|
|
||||||
// 내가 방장도 아니고(roomMasterYn != 'Y'),
|
|
||||||
// 방이 진행중/종료가 아닌 상태면 => "강퇴"로 간주
|
|
||||||
if (!amIStillHere && !_kickedOut && roomMasterYn != 'Y') {
|
if (!amIStillHere && !_kickedOut && roomMasterYn != 'Y') {
|
||||||
// ※ 방이 이미 RUNNING이면 그냥 게임중에 퇴장인지,
|
_kickedOut = true;
|
||||||
// 방이 DELETE 상태인지 등 필요 시 조건 보강
|
|
||||||
|
|
||||||
_kickedOut = true; // 중복 안내 막기
|
|
||||||
|
|
||||||
// (★) 강퇴 안내 + 메인으로 이동
|
|
||||||
Future.delayed(Duration.zero, () async {
|
Future.delayed(Duration.zero, () async {
|
||||||
await showResponseDialog(context, '안내', '강퇴되었습니다.');
|
await showResponseDialog(context, '안내', '강퇴되었습니다.');
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
@ -166,7 +179,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, onError: (error) {
|
}, onError: (error) {
|
||||||
print('FRD onError: $error');
|
|
||||||
setState(() {
|
setState(() {
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
roomTitle = '오류 발생';
|
roomTitle = '오류 발생';
|
||||||
@ -174,13 +186,64 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 1시간 카운트다운 타이머
|
||||||
|
void _startCountdownTimer() {
|
||||||
|
if (_countdownTimer != null && _countdownTimer!.isActive) {
|
||||||
|
return; // 이미 실행중이면 중복 실행 방지
|
||||||
|
}
|
||||||
|
if (_createDt == null) return;
|
||||||
|
|
||||||
|
_countdownTimer = Timer.periodic(const Duration(seconds: 1), (timer) {
|
||||||
|
// 목표시각: createDt + 1시간
|
||||||
|
final endTime = _createDt!.add(const Duration(hours: 1));
|
||||||
|
final now = DateTime.now();
|
||||||
|
final diff = endTime.difference(now);
|
||||||
|
|
||||||
|
if (diff.isNegative) {
|
||||||
|
// 이미 시간이 지남 -> 자동 종료 로직
|
||||||
|
timer.cancel();
|
||||||
|
_remaining = const Duration(seconds: 0);
|
||||||
|
_onAutoTimeout();
|
||||||
|
} else {
|
||||||
|
setState(() {
|
||||||
|
_remaining = diff;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 1시간 만료 후 자동 종료
|
||||||
|
void _onAutoTimeout() {
|
||||||
|
// 방장 => 방 삭제 (leave API)
|
||||||
|
// 일반 => 그냥 나가기
|
||||||
|
if (roomMasterYn == 'Y') {
|
||||||
|
_requestLeaveRoom();
|
||||||
|
} else {
|
||||||
|
_requestLeaveRoom();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _requestLeaveRoom() async {
|
||||||
|
try {
|
||||||
|
final reqBody = {"room_seq": "${widget.roomSeq}"};
|
||||||
|
final response = await Api.serverRequest(uri: '/room/score/game/leave', body: reqBody);
|
||||||
|
// result ok -> 메인
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
if (mounted) {
|
||||||
|
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
_roomStreamSubscription?.cancel(); // ← 구독 해제
|
_countdownTimer?.cancel();
|
||||||
|
_roomStreamSubscription?.cancel();
|
||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (B) 뒤로가기 -> 방 나가기
|
/// 뒤로가기 → 방 나가기
|
||||||
Future<void> _onLeaveRoom() async {
|
Future<void> _onLeaveRoom() async {
|
||||||
if (roomMasterYn == 'Y') {
|
if (roomMasterYn == 'Y') {
|
||||||
// 방장
|
// 방장
|
||||||
@ -195,15 +258,9 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
side: const BorderSide(color: Colors.black, width: 2),
|
side: const BorderSide(color: Colors.black, width: 2),
|
||||||
),
|
),
|
||||||
title: const Center(
|
title: const Center(
|
||||||
child: Text(
|
child: Text('방 나가기', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||||
'방 나가기',
|
|
||||||
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold, color: Colors.black),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
content: const Text(
|
|
||||||
'방장이 나가면 방은 삭제됩니다.\n정말 나가시겠습니까?',
|
|
||||||
style: TextStyle(fontSize: 14, color: Colors.black),
|
|
||||||
),
|
),
|
||||||
|
content: const Text('방장이 나가면 방은 삭제됩니다.\n정말 나가시겠습니까?', style: TextStyle(fontSize: 14)),
|
||||||
actionsAlignment: MainAxisAlignment.spaceEvenly,
|
actionsAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
@ -211,17 +268,19 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 12),
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
),
|
),
|
||||||
child: const Text('취소'),
|
child: const Text('취소'),
|
||||||
),
|
),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () => Navigator.pop(context, true),
|
onPressed: () {
|
||||||
|
final myUserRef = _roomRef.child('userInfo').child(mySeq);
|
||||||
|
myUserRef.onDisconnect().cancel();
|
||||||
|
Navigator.pop(context, true);
|
||||||
|
},
|
||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 12),
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
),
|
),
|
||||||
child: const Text('확인'),
|
child: const Text('확인'),
|
||||||
@ -233,75 +292,10 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
if (confirm != true) return;
|
if (confirm != true) return;
|
||||||
|
|
||||||
// leave API
|
// leave API
|
||||||
try {
|
await _requestLeaveRoom();
|
||||||
final reqBody = {"room_seq": "${widget.roomSeq}"};
|
|
||||||
final response = await Api.serverRequest(uri: '/room/score/game/leave', body: reqBody);
|
|
||||||
if (response['result'] == 'OK') {
|
|
||||||
final resp = response['response'] ?? {};
|
|
||||||
if (resp['result'] == 'OK') {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
} else {
|
|
||||||
final msg = resp['response_info']?['msg_content'] ?? '방 나가기 실패';
|
|
||||||
final again = await showYesNoDialog(
|
|
||||||
context: context,
|
|
||||||
title: '오류',
|
|
||||||
message: '$msg\n그래도 나가시겠습니까?',
|
|
||||||
yesNo: true,
|
|
||||||
);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
final again = await showYesNoDialog(
|
|
||||||
context: context,
|
|
||||||
title: '오류',
|
|
||||||
message: '서버오류\n그래도 나가시겠습니까?',
|
|
||||||
yesNo: true,
|
|
||||||
);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
final again = await showYesNoDialog(
|
|
||||||
context: context,
|
|
||||||
title: '오류',
|
|
||||||
message: '$e\n그래도 나가시겠습니까?',
|
|
||||||
yesNo: true,
|
|
||||||
);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// 일반
|
// 일반
|
||||||
try {
|
await _requestLeaveRoom();
|
||||||
final reqBody = {"room_seq": "${widget.roomSeq}"};
|
|
||||||
final response = await Api.serverRequest(uri: '/room/score/game/leave', body: reqBody);
|
|
||||||
if (response['result'] == 'OK') {
|
|
||||||
final resp = response['response'] ?? {};
|
|
||||||
if (resp['result'] == 'OK') {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
} else {
|
|
||||||
final msg = resp['response_info']?['msg_content'] ?? '나가기 실패';
|
|
||||||
final again = await showYesNoDialog(context: context, title: '오류', message: msg, yesNo: true);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
final again = await showYesNoDialog(context: context, title: '오류', message: '서버 통신 오류', yesNo: true);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
final again = await showYesNoDialog(context: context, title: '오류', message: '$e', yesNo: true);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -314,17 +308,14 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
return defaultVal;
|
return defaultVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 상단 버튼 (방장=3개, 일반=2개)
|
/// 상단 버튼
|
||||||
Widget _buildTopButtons() {
|
Widget _buildTopButtons() {
|
||||||
if (_isLoading) return const SizedBox();
|
if (_isLoading) return const SizedBox();
|
||||||
|
|
||||||
final me = _userList.firstWhere(
|
final me = _userList.firstWhere((u) => (u['user_seq'] ?? '0') == mySeq, orElse: () => {});
|
||||||
(u) => (u['user_seq'] ?? '0') == mySeq,
|
|
||||||
orElse: () => {},
|
|
||||||
);
|
|
||||||
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
||||||
final bool isReady = (myReadyYn == 'Y');
|
final bool isReady = (myReadyYn == 'Y');
|
||||||
final String readyLabel = isReady ? '준비완료' : '준비';
|
final readyLabel = isReady ? '준비완료' : '준비';
|
||||||
|
|
||||||
final btnStyle = ElevatedButton.styleFrom(
|
final btnStyle = ElevatedButton.styleFrom(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
@ -335,7 +326,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
if (roomMasterYn == 'Y') {
|
if (roomMasterYn == 'Y') {
|
||||||
// 방장 => 3개
|
// 방장 => 3개
|
||||||
return Row(
|
return Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Container(
|
child: Container(
|
||||||
@ -372,7 +362,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
} else {
|
} else {
|
||||||
// 일반 => 2개
|
// 일반 => 2개
|
||||||
return Row(
|
return Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Container(
|
child: Container(
|
||||||
@ -399,15 +388,11 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// READY 토글
|
|
||||||
Future<void> _onToggleReady() async {
|
Future<void> _onToggleReady() async {
|
||||||
try {
|
try {
|
||||||
final me = _userList.firstWhere(
|
final me = _userList.firstWhere((u) => (u['user_seq'] ?? '0') == mySeq, orElse: () => {});
|
||||||
(u) => (u['user_seq'] ?? '0') == mySeq,
|
|
||||||
orElse: () => {},
|
|
||||||
);
|
|
||||||
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
||||||
final bool isReady = (myReadyYn == 'Y');
|
final isReady = (myReadyYn == 'Y');
|
||||||
final newYn = isReady ? 'N' : 'Y';
|
final newYn = isReady ? 'N' : 'Y';
|
||||||
|
|
||||||
final userRef = _roomRef.child('userInfo').child(mySeq);
|
final userRef = _roomRef.child('userInfo').child(mySeq);
|
||||||
@ -417,7 +402,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 방 설정
|
|
||||||
Future<void> _onOpenRoomSetting() async {
|
Future<void> _onOpenRoomSetting() async {
|
||||||
final roomInfo = {
|
final roomInfo = {
|
||||||
"room_seq": "${widget.roomSeq}",
|
"room_seq": "${widget.roomSeq}",
|
||||||
@ -442,12 +426,8 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 게임 시작
|
|
||||||
Future<void> _onGameStart() async {
|
Future<void> _onGameStart() async {
|
||||||
final notReady = _userList.any((u) {
|
final notReady = _userList.any((u) => (u['ready_yn'] ?? 'N').toString().toUpperCase() != 'Y');
|
||||||
final ry = (u['ready_yn'] ?? 'N').toString().toUpperCase();
|
|
||||||
return (ry != 'Y');
|
|
||||||
});
|
|
||||||
if (notReady) {
|
if (notReady) {
|
||||||
showResponseDialog(context, '안내', 'READY되지 않은 참가자가 있습니다(방장 포함).');
|
showResponseDialog(context, '안내', 'READY되지 않은 참가자가 있습니다(방장 포함).');
|
||||||
return;
|
return;
|
||||||
@ -457,37 +437,45 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
"room_seq": "${widget.roomSeq}",
|
"room_seq": "${widget.roomSeq}",
|
||||||
"room_type": "PRIVATE",
|
"room_type": "PRIVATE",
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final response = await Api.serverRequest(
|
final response = await Api.serverRequest(uri: '/room/score/game/start', body: requestBody);
|
||||||
uri: '/room/score/game/start',
|
|
||||||
body: requestBody,
|
|
||||||
);
|
|
||||||
if (response['result'] == 'OK') {
|
if (response['result'] == 'OK') {
|
||||||
final resp = response['response'] ?? {};
|
final resp = response['response'] ?? {};
|
||||||
if (resp['result'] == 'OK') {
|
if (resp['result'] == 'OK') {
|
||||||
print('게임 시작 요청 성공(개인전)');
|
print('게임 시작 요청 성공(개인전)');
|
||||||
} else {
|
} else {
|
||||||
final msgTitle = resp['response_info']?['msg_title'] ?? '오류';
|
// ...
|
||||||
final msgContent = resp['response_info']?['msg_content'] ?? '게임 시작 실패';
|
|
||||||
showResponseDialog(context, msgTitle, msgContent);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
showResponseDialog(context, '실패', '서버 통신 오류');
|
// ...
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
showResponseDialog(context, '오류', '$e');
|
// ...
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// (★) 카운트다운 표시용
|
||||||
|
String _formatDuration(Duration d) {
|
||||||
|
final mm = d.inMinutes.remainder(60).toString().padLeft(2, '0');
|
||||||
|
final ss = d.inSeconds.remainder(60).toString().padLeft(2, '0');
|
||||||
|
return '$mm:$ss';
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
// 남은시간 (기본: 60:00 ~ 0:00)
|
||||||
|
final countdownStr = _formatDuration(_remaining);
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
title: const Text('대기 방 (개인전)', style: TextStyle(color: Colors.white)),
|
// 방 제목 + 남은시간 표시
|
||||||
|
title: Text(
|
||||||
|
(roomTitle.isNotEmpty ? roomTitle : '방 제목') + ' [$countdownStr]',
|
||||||
|
style: const TextStyle(color: Colors.white),
|
||||||
|
),
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
||||||
onPressed: _onLeaveRoom,
|
onPressed: _onLeaveRoom,
|
||||||
@ -510,24 +498,16 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Center(
|
// 상단 버튼들
|
||||||
child: Text(
|
|
||||||
roomTitle.isNotEmpty ? roomTitle : '방 제목',
|
|
||||||
style: const TextStyle(fontSize: 24, fontWeight: FontWeight.bold, color: Colors.black),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
|
|
||||||
_buildTopButtons(),
|
_buildTopButtons(),
|
||||||
const SizedBox(height: 20),
|
const SizedBox(height: 20),
|
||||||
|
|
||||||
const Text('사회자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.black)),
|
const Text('사회자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 8),
|
const SizedBox(height: 8),
|
||||||
_buildAdminSection(),
|
_buildAdminSection(),
|
||||||
const SizedBox(height: 20),
|
const SizedBox(height: 20),
|
||||||
|
|
||||||
const Text('참가자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.black)),
|
const Text('참가자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold)),
|
||||||
const SizedBox(height: 8),
|
const SizedBox(height: 8),
|
||||||
_buildPlayerSection(),
|
_buildPlayerSection(),
|
||||||
],
|
],
|
||||||
@ -536,7 +516,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 사회자
|
|
||||||
Widget _buildAdminSection() {
|
Widget _buildAdminSection() {
|
||||||
final adminList = _userList.where((u) {
|
final adminList = _userList.where((u) {
|
||||||
final t = (u['participant_type'] ?? '').toString().toUpperCase();
|
final t = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
@ -544,7 +523,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
}).toList();
|
}).toList();
|
||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
width: double.infinity,
|
|
||||||
padding: const EdgeInsets.all(8),
|
padding: const EdgeInsets.all(8),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
@ -552,7 +530,7 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
borderRadius: BorderRadius.circular(8),
|
borderRadius: BorderRadius.circular(8),
|
||||||
),
|
),
|
||||||
child: adminList.isEmpty
|
child: adminList.isEmpty
|
||||||
? const Text('사회자가 없습니다.', style: TextStyle(color: Colors.black))
|
? const Text('사회자가 없습니다.')
|
||||||
: Wrap(
|
: Wrap(
|
||||||
spacing: 16,
|
spacing: 16,
|
||||||
runSpacing: 8,
|
runSpacing: 8,
|
||||||
@ -561,7 +539,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 일반 참가자
|
|
||||||
Widget _buildPlayerSection() {
|
Widget _buildPlayerSection() {
|
||||||
final playerList = _userList.where((u) {
|
final playerList = _userList.where((u) {
|
||||||
final t = (u['participant_type'] ?? '').toString().toUpperCase();
|
final t = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
@ -569,7 +546,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
}).toList();
|
}).toList();
|
||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
width: double.infinity,
|
|
||||||
padding: const EdgeInsets.all(8),
|
padding: const EdgeInsets.all(8),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
@ -577,7 +553,7 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
borderRadius: BorderRadius.circular(8),
|
borderRadius: BorderRadius.circular(8),
|
||||||
),
|
),
|
||||||
child: playerList.isEmpty
|
child: playerList.isEmpty
|
||||||
? const Text('참가자가 없습니다.', style: TextStyle(color: Colors.black))
|
? const Text('참가자가 없습니다.')
|
||||||
: SingleChildScrollView(
|
: SingleChildScrollView(
|
||||||
child: Wrap(
|
child: Wrap(
|
||||||
spacing: 16,
|
spacing: 16,
|
||||||
@ -589,13 +565,14 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Seat
|
|
||||||
Widget _buildSeat(Map<String, dynamic> userData) {
|
Widget _buildSeat(Map<String, dynamic> userData) {
|
||||||
final userName = userData['nickname'] ?? '유저';
|
final userName = userData['nickname'] ?? '유저';
|
||||||
final profileImg = userData['profile_img'] ?? '';
|
final profileImg = userData['profile_img'] ?? '';
|
||||||
final readyYn = userData['ready_yn'] ?? 'N';
|
final readyYn = (userData['ready_yn'] ?? 'N').toString().toUpperCase();
|
||||||
final isReady = (readyYn == 'Y');
|
final connectYn = (userData['connect_yn'] ?? 'Y').toString().toUpperCase();
|
||||||
final isMaster = (roomMasterYn == 'Y');
|
final bool isReady = (readyYn == 'Y');
|
||||||
|
final bool isDisconnected = (connectYn == 'N');
|
||||||
|
final bool isMaster = (roomMasterYn == 'Y');
|
||||||
|
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
@ -616,7 +593,6 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
child: Container(
|
child: Container(
|
||||||
margin: const EdgeInsets.only(right: 8),
|
margin: const EdgeInsets.only(right: 8),
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
children: [
|
||||||
Container(
|
Container(
|
||||||
width: 50,
|
width: 50,
|
||||||
@ -624,40 +600,55 @@ class _WaitingRoomPrivatePageState extends State<WaitingRoomPrivatePage> {
|
|||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
border: Border.all(
|
border: Border.all(
|
||||||
color: isReady ? Colors.red : Colors.black,
|
color: isDisconnected
|
||||||
width: isReady ? 2 : 1,
|
? Colors.orange
|
||||||
|
: (isReady ? Colors.red : Colors.black),
|
||||||
|
width: isDisconnected ? 2 : (isReady ? 2 : 1),
|
||||||
),
|
),
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20),
|
||||||
boxShadow: isReady
|
boxShadow: [
|
||||||
? [
|
if (isReady)
|
||||||
BoxShadow(
|
BoxShadow(
|
||||||
color: Colors.redAccent.withOpacity(0.6),
|
color: Colors.redAccent.withOpacity(0.6),
|
||||||
blurRadius: 8,
|
blurRadius: 8,
|
||||||
spreadRadius: 2,
|
spreadRadius: 2,
|
||||||
offset: const Offset(0, 0),
|
),
|
||||||
)
|
if (isDisconnected)
|
||||||
]
|
BoxShadow(
|
||||||
: [],
|
color: Colors.orangeAccent.withOpacity(0.6),
|
||||||
|
blurRadius: 8,
|
||||||
|
spreadRadius: 2,
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
child: ClipRRect(
|
child: ClipRRect(
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20),
|
||||||
child: Image.network(
|
child: isDisconnected
|
||||||
|
? const Center(
|
||||||
|
child: Text(
|
||||||
|
'!',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 20,
|
||||||
|
color: Colors.orange,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: Image.network(
|
||||||
'https://eldsoft.com:8097/images$profileImg',
|
'https://eldsoft.com:8097/images$profileImg',
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
errorBuilder: (ctx, err, st) {
|
errorBuilder: (_, __, ___) => const Center(
|
||||||
return const Center(
|
|
||||||
child: Text(
|
child: Text(
|
||||||
'이미지\n불가',
|
'이미지\n불가',
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
style: TextStyle(fontSize: 10),
|
style: TextStyle(fontSize: 10),
|
||||||
),
|
),
|
||||||
);
|
),
|
||||||
},
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 4),
|
const SizedBox(height: 4),
|
||||||
Text(userName, style: const TextStyle(fontSize: 12, color: Colors.black)),
|
Text(userName, style: const TextStyle(fontSize: 12)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
@ -4,9 +4,9 @@ import 'package:firebase_database/firebase_database.dart';
|
|||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
import 'main_page.dart';
|
import 'main_page.dart';
|
||||||
import '../../plugins/api.dart'; // 서버 API
|
import '../../plugins/api.dart';
|
||||||
import '../../dialogs/response_dialog.dart'; // 응답 모달
|
import '../../dialogs/response_dialog.dart';
|
||||||
import '../../dialogs/yes_no_dialog.dart'; // 예/아니오 모달
|
import '../../dialogs/yes_no_dialog.dart';
|
||||||
import '../../dialogs/room_setting_dialog.dart';
|
import '../../dialogs/room_setting_dialog.dart';
|
||||||
import '../../dialogs/user_info_team_dialog.dart';
|
import '../../dialogs/user_info_team_dialog.dart';
|
||||||
import '../../dialogs/team_name_edit_dialog.dart';
|
import '../../dialogs/team_name_edit_dialog.dart';
|
||||||
@ -27,9 +27,7 @@ class WaitingRoomTeamPage extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
||||||
// ─────────────────────────────────────────
|
|
||||||
// 방 설정
|
// 방 설정
|
||||||
// ─────────────────────────────────────────
|
|
||||||
String roomMasterYn = 'N';
|
String roomMasterYn = 'N';
|
||||||
String roomTitle = '';
|
String roomTitle = '';
|
||||||
String roomIntro = '';
|
String roomIntro = '';
|
||||||
@ -40,57 +38,57 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
String scoreOpenRange = 'PRIVATE';
|
String scoreOpenRange = 'PRIVATE';
|
||||||
int numberOfTeams = 1;
|
int numberOfTeams = 1;
|
||||||
|
|
||||||
// 팀명 리스트
|
|
||||||
List<String> _teamNameList = [];
|
List<String> _teamNameList = [];
|
||||||
|
|
||||||
// 유저 목록
|
|
||||||
List<Map<String, dynamic>> _userList = [];
|
List<Map<String, dynamic>> _userList = [];
|
||||||
|
|
||||||
bool _isLoading = true;
|
bool _isLoading = true;
|
||||||
|
|
||||||
// FRD
|
|
||||||
late DatabaseReference _roomRef;
|
late DatabaseReference _roomRef;
|
||||||
Stream<DatabaseEvent>? _roomStream;
|
Stream<DatabaseEvent>? _roomStream;
|
||||||
|
|
||||||
// 진행중 화면 중복 이동 방지
|
|
||||||
bool _movedToRunningPage = false;
|
|
||||||
|
|
||||||
// 강퇴 안내 중복 방지
|
|
||||||
bool _kickedOut = false;
|
|
||||||
|
|
||||||
// FRD 구독 해제
|
|
||||||
StreamSubscription<DatabaseEvent>? _roomStreamSubscription;
|
StreamSubscription<DatabaseEvent>? _roomStreamSubscription;
|
||||||
|
|
||||||
// 로컬스토리지에서 가져올 user_seq
|
bool _movedToRunningPage = false;
|
||||||
String mySeq = '0'; // 원래 '6'을 하드코딩 했던 부분을 제거
|
bool _kickedOut = false;
|
||||||
|
|
||||||
|
String mySeq = '0';
|
||||||
|
|
||||||
|
// (★) 1시간 카운트다운
|
||||||
|
Timer? _countdownTimer;
|
||||||
|
Duration _remaining = const Duration(hours: 1);
|
||||||
|
DateTime? _createDt;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_loadMySeq();
|
FirebaseDatabase.instance.goOnline();
|
||||||
|
_initRoomRef();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// (A) 내 user_seq를 로드하고 나서 방 레퍼런스 설정 + 리스너 등록
|
Future<void> _initRoomRef() async {
|
||||||
Future<void> _loadMySeq() async {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
// 예: 저장된 자료형에 따라 getString or getInt
|
|
||||||
mySeq = prefs.getInt('my_user_seq')?.toString() ?? '0';
|
mySeq = prefs.getInt('my_user_seq')?.toString() ?? '0';
|
||||||
// roomKey / FRD 설정
|
|
||||||
final roomKey = 'korea-${widget.roomSeq}';
|
final roomKey = 'korea-${widget.roomSeq}';
|
||||||
_roomRef = FirebaseDatabase.instance.ref('rooms/$roomKey');
|
_roomRef = FirebaseDatabase.instance.ref('rooms/$roomKey');
|
||||||
// 리스너 시작
|
|
||||||
|
// onDisconnect + connect_yn='Y'
|
||||||
|
final myUserRef = _roomRef.child('userInfo').child(mySeq);
|
||||||
|
myUserRef.onDisconnect().update({'connect_yn': 'N'});
|
||||||
|
await myUserRef.update({'connect_yn': 'Y'});
|
||||||
|
|
||||||
_listenRoomData();
|
_listenRoomData();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
_roomStreamSubscription?.cancel(); // ← 구독 해제
|
_countdownTimer?.cancel();
|
||||||
|
_roomStreamSubscription?.cancel();
|
||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _listenRoomData() {
|
void _listenRoomData() {
|
||||||
_roomStream = _roomRef.onValue;
|
_roomStream = _roomRef.onValue;
|
||||||
_roomStream?.listen((event) async {
|
_roomStreamSubscription = _roomStream?.listen((event) {
|
||||||
final snapshot = event.snapshot;
|
final snapshot = event.snapshot;
|
||||||
if (!snapshot.exists) {
|
if (!snapshot.exists) {
|
||||||
setState(() {
|
setState(() {
|
||||||
@ -105,7 +103,6 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
final roomInfoData = data['roomInfo'] as Map<dynamic, dynamic>? ?? {};
|
final roomInfoData = data['roomInfo'] as Map<dynamic, dynamic>? ?? {};
|
||||||
final userInfoData = data['userInfo'] as Map<dynamic, dynamic>? ?? {};
|
final userInfoData = data['userInfo'] as Map<dynamic, dynamic>? ?? {};
|
||||||
|
|
||||||
// 현재 방 상태
|
|
||||||
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
final roomStatus = (roomInfoData['room_status'] ?? 'WAIT').toString().toUpperCase();
|
||||||
|
|
||||||
setState(() {
|
setState(() {
|
||||||
@ -118,7 +115,7 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
scoreOpenRange = (roomInfoData['score_open_range'] ?? 'PRIVATE') as String;
|
scoreOpenRange = (roomInfoData['score_open_range'] ?? 'PRIVATE') as String;
|
||||||
numberOfTeams = _toInt(roomInfoData['number_of_teams'], 1);
|
numberOfTeams = _toInt(roomInfoData['number_of_teams'], 1);
|
||||||
|
|
||||||
// 팀명 리스트
|
// 팀명
|
||||||
final tStr = (roomInfoData['team_name_list'] ?? '') as String;
|
final tStr = (roomInfoData['team_name_list'] ?? '') as String;
|
||||||
if (tStr.isNotEmpty) {
|
if (tStr.isNotEmpty) {
|
||||||
_teamNameList = tStr.split(',').map((e) => e.trim().toUpperCase()).toList();
|
_teamNameList = tStr.split(',').map((e) => e.trim().toUpperCase()).toList();
|
||||||
@ -126,14 +123,14 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
_teamNameList = List.generate(numberOfTeams, (i) => String.fromCharCode(65 + i));
|
_teamNameList = List.generate(numberOfTeams, (i) => String.fromCharCode(65 + i));
|
||||||
}
|
}
|
||||||
|
|
||||||
// 방장 여부
|
// 방장
|
||||||
roomMasterYn = 'N';
|
roomMasterYn = 'N';
|
||||||
final masterSeq = roomInfoData['master_user_seq'];
|
final masterSeq = roomInfoData['master_user_seq'];
|
||||||
if (masterSeq != null && masterSeq.toString() == mySeq) {
|
if (masterSeq != null && masterSeq.toString() == mySeq) {
|
||||||
roomMasterYn = 'Y';
|
roomMasterYn = 'Y';
|
||||||
}
|
}
|
||||||
|
|
||||||
// 유저 목록
|
// userList
|
||||||
final tempList = <Map<String, dynamic>>[];
|
final tempList = <Map<String, dynamic>>[];
|
||||||
userInfoData.forEach((userSeq, userMap) {
|
userInfoData.forEach((userSeq, userMap) {
|
||||||
tempList.add({
|
tempList.add({
|
||||||
@ -143,16 +140,16 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
'team_name': userMap['team_name'] ?? '',
|
'team_name': userMap['team_name'] ?? '',
|
||||||
'score': userMap['score'] ?? 0,
|
'score': userMap['score'] ?? 0,
|
||||||
'profile_img': userMap['profile_img'] ?? '',
|
'profile_img': userMap['profile_img'] ?? '',
|
||||||
'department': userMap['department'] ?? '',
|
|
||||||
'introduce_myself': userMap['introduce_myself'] ?? '',
|
'introduce_myself': userMap['introduce_myself'] ?? '',
|
||||||
'ready_yn': (userMap['ready_yn'] ?? 'N').toString().toUpperCase(),
|
'ready_yn': (userMap['ready_yn'] ?? 'N').toString().toUpperCase(),
|
||||||
|
'connect_yn': (userMap['connect_yn'] ?? 'Y').toString().toUpperCase(),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
_userList = tempList;
|
_userList = tempList;
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
});
|
});
|
||||||
|
|
||||||
// 상태가 RUNNING이면 진행중 화면으로
|
// 진행중 -> 이동
|
||||||
if (roomStatus == 'RUNNING' && !_movedToRunningPage) {
|
if (roomStatus == 'RUNNING' && !_movedToRunningPage) {
|
||||||
_movedToRunningPage = true;
|
_movedToRunningPage = true;
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
@ -167,20 +164,20 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// (2) 여기서 "내 user_seq가 목록에 있는지" 검사
|
// (C) create_dt 파싱 -> 1시간 카운트다운
|
||||||
|
final createDtStr = (roomInfoData['create_dt'] ?? '') as String;
|
||||||
|
if (createDtStr.isNotEmpty && createDtStr.contains('T')) {
|
||||||
|
final dt = DateTime.tryParse(createDtStr);
|
||||||
|
if (dt != null) {
|
||||||
|
_createDt = dt;
|
||||||
|
_startCountdownTimer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 강퇴판별
|
||||||
final amIStillHere = _userList.any((u) => u['user_seq'].toString() == mySeq);
|
final amIStillHere = _userList.any((u) => u['user_seq'].toString() == mySeq);
|
||||||
|
|
||||||
// (3) 만약 내가 목록에서 사라졌고,
|
|
||||||
// 아직 안내하지 않았으며(_kickedOut == false),
|
|
||||||
// 내가 방장도 아니고(roomMasterYn != 'Y'),
|
|
||||||
// 방이 진행중/종료가 아닌 상태면 => "강퇴"로 간주
|
|
||||||
if (!amIStillHere && !_kickedOut && roomMasterYn != 'Y') {
|
if (!amIStillHere && !_kickedOut && roomMasterYn != 'Y') {
|
||||||
// ※ 방이 이미 RUNNING이면 그냥 게임중에 퇴장인지,
|
_kickedOut = true;
|
||||||
// 방이 DELETE 상태인지 등 필요 시 조건 보강
|
|
||||||
|
|
||||||
_kickedOut = true; // 중복 안내 막기
|
|
||||||
|
|
||||||
// (★) 강퇴 안내 + 메인으로 이동
|
|
||||||
Future.delayed(Duration.zero, () async {
|
Future.delayed(Duration.zero, () async {
|
||||||
await showResponseDialog(context, '안내', '강퇴되었습니다.');
|
await showResponseDialog(context, '안내', '강퇴되었습니다.');
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
@ -190,7 +187,6 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, onError: (error) {
|
}, onError: (error) {
|
||||||
print('FRD onError: $error');
|
|
||||||
setState(() {
|
setState(() {
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
roomTitle = '오류 발생';
|
roomTitle = '오류 발생';
|
||||||
@ -198,12 +194,51 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// ─────────────────────────────────────────
|
// 카운트다운
|
||||||
// [추가] 뒤로가기 -> 방 나가기
|
void _startCountdownTimer() {
|
||||||
// ─────────────────────────────────────────
|
if (_countdownTimer != null && _countdownTimer!.isActive) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (_createDt == null) return;
|
||||||
|
|
||||||
|
_countdownTimer = Timer.periodic(const Duration(seconds: 1), (timer) {
|
||||||
|
final endTime = _createDt!.add(const Duration(hours: 1));
|
||||||
|
final now = DateTime.now();
|
||||||
|
final diff = endTime.difference(now);
|
||||||
|
|
||||||
|
if (diff.isNegative) {
|
||||||
|
timer.cancel();
|
||||||
|
_remaining = const Duration(seconds: 0);
|
||||||
|
_onAutoTimeout();
|
||||||
|
} else {
|
||||||
|
setState(() {
|
||||||
|
_remaining = diff;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onAutoTimeout() {
|
||||||
|
// 자동 종료 -> 방장=나가기(방삭제), 일반=나가기
|
||||||
|
_requestLeaveRoom();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _requestLeaveRoom() async {
|
||||||
|
try {
|
||||||
|
final reqBody = {"room_seq": "${widget.roomSeq}"};
|
||||||
|
final response = await Api.serverRequest(uri: '/room/score/game/leave', body: reqBody);
|
||||||
|
// ...
|
||||||
|
} catch (e) {
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
if (mounted) {
|
||||||
|
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 뒤로가기 -> 방 나가기
|
||||||
Future<void> _onLeaveRoom() async {
|
Future<void> _onLeaveRoom() async {
|
||||||
if (roomMasterYn == 'Y') {
|
if (roomMasterYn == 'Y') {
|
||||||
// 방장 -> 경고 모달
|
|
||||||
final confirm = await showDialog<bool>(
|
final confirm = await showDialog<bool>(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
barrierDismissible: false,
|
||||||
@ -215,15 +250,9 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
side: const BorderSide(color: Colors.black, width: 2),
|
side: const BorderSide(color: Colors.black, width: 2),
|
||||||
),
|
),
|
||||||
title: const Center(
|
title: const Center(
|
||||||
child: Text(
|
child: Text('방 나가기', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold, color: Colors.black)),
|
||||||
'방 나가기',
|
|
||||||
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold, color: Colors.black),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
content: const Text(
|
|
||||||
'방장이 나가면 방은 삭제됩니다.\n정말 나가시겠습니까?',
|
|
||||||
style: TextStyle(fontSize: 14, color: Colors.black),
|
|
||||||
),
|
),
|
||||||
|
content: const Text('방장이 나가면 방은 삭제됩니다.\n정말 나가시겠습니까?', style: TextStyle(fontSize: 14)),
|
||||||
actionsAlignment: MainAxisAlignment.spaceEvenly,
|
actionsAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
@ -231,17 +260,19 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 12),
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
),
|
),
|
||||||
child: const Text('취소'),
|
child: const Text('취소'),
|
||||||
),
|
),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () => Navigator.pop(context, true),
|
onPressed: () {
|
||||||
|
final myUserRef = _roomRef.child('userInfo').child(mySeq);
|
||||||
|
myUserRef.onDisconnect().cancel();
|
||||||
|
Navigator.pop(context, true);
|
||||||
|
},
|
||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 12),
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||||
),
|
),
|
||||||
child: const Text('확인'),
|
child: const Text('확인'),
|
||||||
@ -252,75 +283,9 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
);
|
);
|
||||||
if (confirm != true) return;
|
if (confirm != true) return;
|
||||||
|
|
||||||
try {
|
await _requestLeaveRoom();
|
||||||
final reqBody = {"room_seq": "${widget.roomSeq}"};
|
|
||||||
final response = await Api.serverRequest(uri: '/room/score/game/leave', body: reqBody);
|
|
||||||
if (response['result'] == 'OK') {
|
|
||||||
final resp = response['response'];
|
|
||||||
if (resp != null && resp['result'] == 'OK') {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
} else {
|
} else {
|
||||||
final msg = resp?['response_info']?['msg_content'] ?? '방 나가기 실패';
|
await _requestLeaveRoom();
|
||||||
final again = await showYesNoDialog(
|
|
||||||
context: context,
|
|
||||||
title: '오류',
|
|
||||||
message: '$msg\n그래도 나가시겠습니까?',
|
|
||||||
yesNo: true,
|
|
||||||
);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
final again = await showYesNoDialog(
|
|
||||||
context: context,
|
|
||||||
title: '오류',
|
|
||||||
message: '서버오류\n그래도 나가시겠습니까?',
|
|
||||||
yesNo: true,
|
|
||||||
);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
final again = await showYesNoDialog(
|
|
||||||
context: context,
|
|
||||||
title: '오류',
|
|
||||||
message: '$e\n그래도 나가시겠습니까?',
|
|
||||||
yesNo: true,
|
|
||||||
);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// 일반 유저
|
|
||||||
try {
|
|
||||||
final reqBody = {"room_seq": "${widget.roomSeq}"};
|
|
||||||
final response = await Api.serverRequest(uri: '/room/score/game/leave', body: reqBody);
|
|
||||||
if (response['result'] == 'OK') {
|
|
||||||
final resp = response['response'] ?? {};
|
|
||||||
if (resp['result'] == 'OK') {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
} else {
|
|
||||||
final msg = resp['response_info']?['msg_content'] ?? '나가기 실패';
|
|
||||||
final again = await showYesNoDialog(context: context, title: '오류', message: msg, yesNo: true);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
final again = await showYesNoDialog(context: context, title: '오류', message: '서버 통신 오류', yesNo: true);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
final again = await showYesNoDialog(context: context, title: '오류', message: '$e', yesNo: true);
|
|
||||||
if (again == true) {
|
|
||||||
Navigator.pushReplacement(context, MaterialPageRoute(builder: (_) => const MainPage()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -333,22 +298,14 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
return defaultVal;
|
return defaultVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ─────────────────────────────────────────
|
// 상단 버튼
|
||||||
// 상단 버튼들: 방장 = 3개, 일반 = 2개
|
|
||||||
// READY 버튼에 "준비"/"준비완료" 표시
|
|
||||||
// 게임 시작: 전체 READY=Y 필요
|
|
||||||
// ─────────────────────────────────────────
|
|
||||||
Widget _buildTopButtons() {
|
Widget _buildTopButtons() {
|
||||||
if (_isLoading) return const SizedBox();
|
if (_isLoading) return const SizedBox();
|
||||||
|
|
||||||
// 내 READY 상태
|
final me = _userList.firstWhere((u) => (u['user_seq'] ?? '0') == mySeq, orElse: () => {});
|
||||||
final me = _userList.firstWhere(
|
|
||||||
(u) => (u['user_seq'] ?? '0') == mySeq,
|
|
||||||
orElse: () => {},
|
|
||||||
);
|
|
||||||
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
||||||
final bool isReady = (myReadyYn == 'Y');
|
final bool isReady = (myReadyYn == 'Y');
|
||||||
final String readyLabel = isReady ? '준비완료' : '준비';
|
final readyLabel = isReady ? '준비완료' : '준비';
|
||||||
|
|
||||||
final btnStyle = ElevatedButton.styleFrom(
|
final btnStyle = ElevatedButton.styleFrom(
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
@ -357,9 +314,7 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (roomMasterYn == 'Y') {
|
if (roomMasterYn == 'Y') {
|
||||||
// 방장 -> [방 설정], [준비/준비완료], [게임 시작]
|
|
||||||
return Row(
|
return Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Container(
|
child: Container(
|
||||||
@ -394,9 +349,7 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
],
|
],
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
// 일반 -> [방 설정], [준비/준비완료]
|
|
||||||
return Row(
|
return Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Container(
|
child: Container(
|
||||||
@ -423,16 +376,11 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// READY 토글
|
|
||||||
Future<void> _onToggleReady() async {
|
Future<void> _onToggleReady() async {
|
||||||
try {
|
try {
|
||||||
// 내 데이터
|
final me = _userList.firstWhere((u) => (u['user_seq'] ?? '') == mySeq, orElse: () => {});
|
||||||
final me = _userList.firstWhere(
|
|
||||||
(u) => (u['user_seq'] ?? '') == mySeq,
|
|
||||||
orElse: () => {},
|
|
||||||
);
|
|
||||||
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
final myReadyYn = (me['ready_yn'] ?? 'N').toString().toUpperCase();
|
||||||
final bool isReady = (myReadyYn == 'Y');
|
final isReady = (myReadyYn == 'Y');
|
||||||
final newYn = isReady ? 'N' : 'Y';
|
final newYn = isReady ? 'N' : 'Y';
|
||||||
|
|
||||||
final userRef = _roomRef.child('userInfo').child(mySeq);
|
final userRef = _roomRef.child('userInfo').child(mySeq);
|
||||||
@ -442,7 +390,6 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 방 설정 열기
|
|
||||||
Future<void> _onOpenRoomSetting() async {
|
Future<void> _onOpenRoomSetting() async {
|
||||||
final roomInfo = {
|
final roomInfo = {
|
||||||
"room_seq": "${widget.roomSeq}",
|
"room_seq": "${widget.roomSeq}",
|
||||||
@ -465,16 +412,12 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
builder: (_) => RoomSettingModal(roomInfo: roomInfo),
|
builder: (_) => RoomSettingModal(roomInfo: roomInfo),
|
||||||
);
|
);
|
||||||
if (result == 'refresh') {
|
if (result == 'refresh') {
|
||||||
// do something
|
// ...
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 게임 시작 (전체 READY=Y 필요)
|
|
||||||
Future<void> _onGameStart() async {
|
Future<void> _onGameStart() async {
|
||||||
final notReady = _userList.any((u) {
|
final notReady = _userList.any((u) => (u['ready_yn'] ?? 'N').toString().toUpperCase() != 'Y');
|
||||||
final ry = (u['ready_yn'] ?? 'N').toString().toUpperCase();
|
|
||||||
return (ry != 'Y');
|
|
||||||
});
|
|
||||||
if (notReady) {
|
if (notReady) {
|
||||||
showResponseDialog(context, '안내', 'READY되지 않은 참가자가 있습니다(방장 포함).');
|
showResponseDialog(context, '안내', 'READY되지 않은 참가자가 있습니다(방장 포함).');
|
||||||
return;
|
return;
|
||||||
@ -485,30 +428,85 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
"room_type": "TEAM",
|
"room_type": "TEAM",
|
||||||
};
|
};
|
||||||
try {
|
try {
|
||||||
final response = await Api.serverRequest(
|
final response = await Api.serverRequest(uri: '/room/score/game/start', body: requestBody);
|
||||||
uri: '/room/score/game/start',
|
|
||||||
body: requestBody,
|
|
||||||
);
|
|
||||||
if (response['result'] == 'OK') {
|
if (response['result'] == 'OK') {
|
||||||
final resp = response['response'] ?? {};
|
final resp = response['response'] ?? {};
|
||||||
if (resp['result'] == 'OK') {
|
if (resp['result'] == 'OK') {
|
||||||
print('게임 시작 요청 성공(팀전)');
|
print('게임 시작 요청 성공(팀전)');
|
||||||
} else {
|
} else {
|
||||||
final msgTitle = resp['response_info']?['msg_title'] ?? '오류';
|
// ...
|
||||||
final msgContent = resp['response_info']?['msg_content'] ?? '게임 시작 실패';
|
|
||||||
showResponseDialog(context, msgTitle, msgContent);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
showResponseDialog(context, '실패', '서버 통신 오류');
|
// ...
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
showResponseDialog(context, '오류', '$e');
|
// ...
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ─────────────────────────────────────────
|
// (★) 카운트다운 표시용
|
||||||
// 사회자 / 팀 섹션 / 대기중 / Seat
|
String _formatDuration(Duration d) {
|
||||||
// ─────────────────────────────────────────
|
final mm = d.inMinutes.remainder(60).toString().padLeft(2, '0');
|
||||||
|
final ss = d.inSeconds.remainder(60).toString().padLeft(2, '0');
|
||||||
|
return '$mm:$ss';
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final countdownStr = _formatDuration(_remaining);
|
||||||
|
|
||||||
|
return Scaffold(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
appBar: AppBar(
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
elevation: 0,
|
||||||
|
// 방 제목 + 남은시간
|
||||||
|
title: Text(
|
||||||
|
(roomTitle.isNotEmpty ? roomTitle : '방 제목') + ' [$countdownStr]',
|
||||||
|
style: const TextStyle(color: Colors.white),
|
||||||
|
),
|
||||||
|
leading: IconButton(
|
||||||
|
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
||||||
|
onPressed: _onLeaveRoom,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
bottomNavigationBar: Container(
|
||||||
|
height: 50,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.grey.shade300,
|
||||||
|
border: Border.all(color: Colors.black, width: 1),
|
||||||
|
),
|
||||||
|
child: const Center(
|
||||||
|
child: Text('구글 광고', style: TextStyle(color: Colors.black)),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
body: _isLoading
|
||||||
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 16, horizontal: 16),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
_buildTopButtons(),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
|
||||||
|
const Text('사회자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.black)),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildAdminSection(),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
|
||||||
|
const Text('팀별 참가자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.black)),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildTeamSection(),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
|
||||||
|
_buildWaitSection(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
Widget _buildAdminSection() {
|
Widget _buildAdminSection() {
|
||||||
final adminList = _userList.where((u) {
|
final adminList = _userList.where((u) {
|
||||||
final pType = (u['participant_type'] ?? '').toString().toUpperCase();
|
final pType = (u['participant_type'] ?? '').toString().toUpperCase();
|
||||||
@ -516,7 +514,6 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
}).toList();
|
}).toList();
|
||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
width: double.infinity,
|
|
||||||
padding: const EdgeInsets.all(8),
|
padding: const EdgeInsets.all(8),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
@ -553,7 +550,6 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
|
|
||||||
if (teamMap.isEmpty) {
|
if (teamMap.isEmpty) {
|
||||||
return Container(
|
return Container(
|
||||||
width: double.infinity,
|
|
||||||
padding: const EdgeInsets.all(8),
|
padding: const EdgeInsets.all(8),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
@ -660,13 +656,14 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
Widget _buildSeat(Map<String, dynamic> user) {
|
Widget _buildSeat(Map<String, dynamic> user) {
|
||||||
final userName = user['nickname'] ?? '유저';
|
final userName = user['nickname'] ?? '유저';
|
||||||
final profileImg = user['profile_img'] ?? '';
|
final profileImg = user['profile_img'] ?? '';
|
||||||
final readyYn = user['ready_yn'] ?? 'N';
|
final readyYn = (user['ready_yn'] ?? 'N').toString().toUpperCase();
|
||||||
final isReady = (readyYn == 'Y');
|
final connectYn = (user['connect_yn'] ?? 'Y').toString().toUpperCase();
|
||||||
final isMaster = (roomMasterYn == 'Y');
|
final bool isReady = (readyYn == 'Y');
|
||||||
|
final bool isDisconnected = (connectYn == 'N');
|
||||||
|
final bool isMaster = (roomMasterYn == 'Y');
|
||||||
|
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
// 유저 정보 모달
|
|
||||||
final result = await showDialog(
|
final result = await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
barrierDismissible: false,
|
||||||
@ -674,7 +671,7 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
userData: user,
|
userData: user,
|
||||||
isRoomMaster: isMaster,
|
isRoomMaster: isMaster,
|
||||||
roomSeq: widget.roomSeq,
|
roomSeq: widget.roomSeq,
|
||||||
roomTypeName: widget.roomType.toUpperCase(), // "TEAM"
|
roomTypeName: widget.roomType.toUpperCase(),
|
||||||
teamNameList: _teamNameList,
|
teamNameList: _teamNameList,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
@ -686,7 +683,6 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
width: 60,
|
width: 60,
|
||||||
margin: const EdgeInsets.only(right: 8),
|
margin: const EdgeInsets.only(right: 8),
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
children: [
|
||||||
Container(
|
Container(
|
||||||
width: 50,
|
width: 50,
|
||||||
@ -694,35 +690,50 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
border: Border.all(
|
border: Border.all(
|
||||||
color: isReady ? Colors.red : Colors.black,
|
color: isDisconnected
|
||||||
width: isReady ? 2 : 1,
|
? Colors.orange
|
||||||
|
: (isReady ? Colors.red : Colors.black),
|
||||||
|
width: isDisconnected ? 2 : (isReady ? 2 : 1),
|
||||||
),
|
),
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20),
|
||||||
boxShadow: isReady
|
boxShadow: [
|
||||||
? [
|
if (isReady)
|
||||||
BoxShadow(
|
BoxShadow(
|
||||||
color: Colors.redAccent.withOpacity(0.6),
|
color: Colors.redAccent.withOpacity(0.6),
|
||||||
blurRadius: 8,
|
blurRadius: 8,
|
||||||
spreadRadius: 2,
|
spreadRadius: 2,
|
||||||
offset: const Offset(0, 0),
|
),
|
||||||
)
|
if (isDisconnected)
|
||||||
]
|
BoxShadow(
|
||||||
: [],
|
color: Colors.orangeAccent.withOpacity(0.6),
|
||||||
|
blurRadius: 8,
|
||||||
|
spreadRadius: 2,
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
child: ClipRRect(
|
child: ClipRRect(
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20),
|
||||||
child: Image.network(
|
child: isDisconnected
|
||||||
|
? const Center(
|
||||||
|
child: Text(
|
||||||
|
'!',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 20,
|
||||||
|
color: Colors.orange,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: Image.network(
|
||||||
'https://eldsoft.com:8097/images$profileImg',
|
'https://eldsoft.com:8097/images$profileImg',
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
errorBuilder: (ctx, err, st) {
|
errorBuilder: (_, __, ___) => const Center(
|
||||||
return const Center(
|
|
||||||
child: Text(
|
child: Text(
|
||||||
'이미지\n불가',
|
'이미지\n불가',
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
style: TextStyle(fontSize: 10),
|
style: TextStyle(fontSize: 10),
|
||||||
),
|
),
|
||||||
);
|
),
|
||||||
},
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -733,63 +744,4 @@ class _WaitingRoomTeamPageState extends State<WaitingRoomTeamPage> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Scaffold(
|
|
||||||
backgroundColor: Colors.white,
|
|
||||||
appBar: AppBar(
|
|
||||||
backgroundColor: Colors.black,
|
|
||||||
elevation: 0,
|
|
||||||
title: const Text('대기 방 (팀전)', style: TextStyle(color: Colors.white)),
|
|
||||||
leading: IconButton(
|
|
||||||
icon: const Icon(Icons.arrow_back_ios, color: Colors.white),
|
|
||||||
onPressed: _onLeaveRoom, // 나가기
|
|
||||||
),
|
|
||||||
),
|
|
||||||
bottomNavigationBar: Container(
|
|
||||||
height: 50,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: Colors.grey.shade300,
|
|
||||||
border: Border.all(color: Colors.black, width: 1),
|
|
||||||
),
|
|
||||||
child: const Center(
|
|
||||||
child: Text('구글 광고', style: TextStyle(color: Colors.black)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
body: _isLoading
|
|
||||||
? const Center(child: CircularProgressIndicator())
|
|
||||||
: SingleChildScrollView(
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 16, horizontal: 16),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Center(
|
|
||||||
child: Text(
|
|
||||||
roomTitle.isNotEmpty ? roomTitle : '방 제목',
|
|
||||||
style: const TextStyle(fontSize: 24, fontWeight: FontWeight.bold, color: Colors.black),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
|
|
||||||
_buildTopButtons(),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
|
|
||||||
const Text('사회자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.black)),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
_buildAdminSection(),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
|
|
||||||
const Text('팀별 참가자', style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.black)),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
_buildTeamSection(),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
|
|
||||||
_buildWaitSection(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ import file_selector_macos
|
|||||||
import firebase_auth
|
import firebase_auth
|
||||||
import firebase_core
|
import firebase_core
|
||||||
import firebase_database
|
import firebase_database
|
||||||
|
import google_sign_in_ios
|
||||||
import shared_preferences_foundation
|
import shared_preferences_foundation
|
||||||
import webview_flutter_wkwebview
|
import webview_flutter_wkwebview
|
||||||
|
|
||||||
@ -17,6 +18,7 @@ func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
|
|||||||
FLTFirebaseAuthPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseAuthPlugin"))
|
FLTFirebaseAuthPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseAuthPlugin"))
|
||||||
FLTFirebaseCorePlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseCorePlugin"))
|
FLTFirebaseCorePlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseCorePlugin"))
|
||||||
FLTFirebaseDatabasePlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseDatabasePlugin"))
|
FLTFirebaseDatabasePlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseDatabasePlugin"))
|
||||||
|
FLTGoogleSignInPlugin.register(with: registry.registrar(forPlugin: "FLTGoogleSignInPlugin"))
|
||||||
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
|
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
|
||||||
FLTWebViewFlutterPlugin.register(with: registry.registrar(forPlugin: "FLTWebViewFlutterPlugin"))
|
FLTWebViewFlutterPlugin.register(with: registry.registrar(forPlugin: "FLTWebViewFlutterPlugin"))
|
||||||
}
|
}
|
||||||
|
BIN
my_release_key.jks
Normal file
64
pubspec.lock
@ -210,10 +210,10 @@ packages:
|
|||||||
dependency: "direct dev"
|
dependency: "direct dev"
|
||||||
description:
|
description:
|
||||||
name: flutter_lints
|
name: flutter_lints
|
||||||
sha256: "5398f14efa795ffb7a33e9b6a08798b26a180edac4ad7db3f231e40f82ce11e1"
|
sha256: a25a15ebbdfc33ab1cd26c63a6ee519df92338a9c10f122adda92938253bef04
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "5.0.0"
|
version: "2.0.3"
|
||||||
flutter_plugin_android_lifecycle:
|
flutter_plugin_android_lifecycle:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -232,6 +232,14 @@ packages:
|
|||||||
description: flutter
|
description: flutter
|
||||||
source: sdk
|
source: sdk
|
||||||
version: "0.0.0"
|
version: "0.0.0"
|
||||||
|
fluttertoast:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: fluttertoast
|
||||||
|
sha256: "24467dc20bbe49fd63e57d8e190798c4d22cbbdac30e54209d153a15273721d1"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "8.2.10"
|
||||||
google_mobile_ads:
|
google_mobile_ads:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@ -240,6 +248,46 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "5.2.0"
|
version: "5.2.0"
|
||||||
|
google_sign_in:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: google_sign_in
|
||||||
|
sha256: "821f354c053d51a2d417b02d42532a19a6ea8057d2f9ebb8863c07d81c98aaf9"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "5.4.4"
|
||||||
|
google_sign_in_android:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: google_sign_in_android
|
||||||
|
sha256: "3b96f9b6cf61915f73cbe1218a192623e296a9b8b31965702503649477761e36"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "6.1.34"
|
||||||
|
google_sign_in_ios:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: google_sign_in_ios
|
||||||
|
sha256: "83f015169102df1ab2905cf8abd8934e28f87db9ace7a5fa676998842fed228a"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "5.7.8"
|
||||||
|
google_sign_in_platform_interface:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: google_sign_in_platform_interface
|
||||||
|
sha256: "1f6e5787d7a120cc0359ddf315c92309069171306242e181c09472d1b00a2971"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.4.5"
|
||||||
|
google_sign_in_web:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: google_sign_in_web
|
||||||
|
sha256: "75cc41ebc53b1756320ee14d9c3018ad3e6cea298147dbcd86e9d0c8d6720b40"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.10.2+1"
|
||||||
http:
|
http:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@ -320,6 +368,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.2.1+1"
|
version: "0.2.1+1"
|
||||||
|
js:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: js
|
||||||
|
sha256: f2c445dce49627136094980615a031419f7f3eb393237e4ecd97ac15dea343f3
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.6.7"
|
||||||
leak_tracker:
|
leak_tracker:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -348,10 +404,10 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: lints
|
name: lints
|
||||||
sha256: c35bb79562d980e9a453fc715854e1ed39e24e7d0297a880ef54e17f9874a9d7
|
sha256: "0a217c6c989d21039f1498c3ed9f3ed71b354e69873f13a8dfc3c9fe76f1b452"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "5.1.1"
|
version: "2.1.1"
|
||||||
matcher:
|
matcher:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
98
pubspec.yaml
@ -1,35 +1,18 @@
|
|||||||
name: allscore_app
|
name: allscore_app
|
||||||
description: "A new Flutter project."
|
description: "A new Flutter project."
|
||||||
# The following line prevents the package from being accidentally published to
|
publish_to: 'none'
|
||||||
# pub.dev using `flutter pub publish`. This is preferred for private packages.
|
|
||||||
publish_to: 'none' # Remove this line if you wish to publish to pub.dev
|
|
||||||
|
|
||||||
# The following defines the version and build number for your application.
|
|
||||||
# A version number is three numbers separated by dots, like 1.2.43
|
|
||||||
# followed by an optional build number separated by a +.
|
|
||||||
# Both the version and the builder number may be overridden in flutter
|
|
||||||
# build by specifying --build-name and --build-number, respectively.
|
|
||||||
# In Android, build-name is used as versionName while build-number used as versionCode.
|
|
||||||
# Read more about Android versioning at https://developer.android.com/studio/publish/versioning
|
|
||||||
# In iOS, build-name is used as CFBundleShortVersionString while build-number is used as CFBundleVersion.
|
|
||||||
# Read more about iOS versioning at
|
|
||||||
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
|
|
||||||
# In Windows, build-name is used as the major, minor, and patch parts
|
|
||||||
# of the product and file versions while build-number is used as the build suffix.
|
|
||||||
version: 1.0.0+1
|
version: 1.0.0+1
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
sdk: ^3.6.0
|
sdk: ^3.6.0
|
||||||
|
|
||||||
# Dependencies specify other packages that your package needs in order to work.
|
# ───────────────────────────────────
|
||||||
# To automatically upgrade your package dependencies to the latest versions
|
# 의존성
|
||||||
# consider running `flutter pub upgrade --major-versions`. Alternatively,
|
# ───────────────────────────────────
|
||||||
# dependencies can be manually updated by changing the version numbers below to
|
|
||||||
# the latest version available on pub.dev. To see which dependencies have newer
|
|
||||||
# versions available, run `flutter pub outdated`.
|
|
||||||
dependencies:
|
dependencies:
|
||||||
flutter:
|
flutter:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
|
|
||||||
google_mobile_ads: ^5.2.0
|
google_mobile_ads: ^5.2.0
|
||||||
http: ^1.2.2
|
http: ^1.2.2
|
||||||
crypto: ^3.0.1
|
crypto: ^3.0.1
|
||||||
@ -38,68 +21,27 @@ dependencies:
|
|||||||
firebase_core: ^3.9.0
|
firebase_core: ^3.9.0
|
||||||
firebase_auth: ^5.3.4
|
firebase_auth: ^5.3.4
|
||||||
firebase_database: ^11.2.0
|
firebase_database: ^11.2.0
|
||||||
|
google_sign_in: ^5.4.0
|
||||||
# The following adds the Cupertino Icons font to your application.
|
|
||||||
# Use with the CupertinoIcons class for iOS style icons.
|
|
||||||
cupertino_icons: ^1.0.8
|
cupertino_icons: ^1.0.8
|
||||||
|
fluttertoast: ^8.0.9
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
google_mobile_ads: ^5.2.0
|
flutter_lints: ^2.0.0
|
||||||
http: ^1.2.2
|
|
||||||
crypto: ^3.0.1
|
|
||||||
shared_preferences: ^2.0.6
|
|
||||||
image_picker: ^0.8.4+4
|
|
||||||
firebase_core: ^3.9.0
|
|
||||||
firebase_auth: ^5.3.4
|
|
||||||
firebase_database: ^11.2.0
|
|
||||||
|
|
||||||
# The "flutter_lints" package below contains a set of recommended lints to
|
# ───────────────────────────────────
|
||||||
# encourage good coding practices. The lint set provided by the package is
|
# 플러터 섹션
|
||||||
# activated in the `analysis_options.yaml` file located at the root of your
|
# ───────────────────────────────────
|
||||||
# package. See that file for information about deactivating specific lint
|
|
||||||
# rules and activating additional ones.
|
|
||||||
flutter_lints: ^5.0.0
|
|
||||||
|
|
||||||
# For information on the generic Dart part of this file, see the
|
|
||||||
# following page: https://dart.dev/tools/pub/pubspec
|
|
||||||
|
|
||||||
# The following section is specific to Flutter packages.
|
|
||||||
flutter:
|
flutter:
|
||||||
|
|
||||||
# The following line ensures that the Material Icons font is
|
|
||||||
# included with your application, so that you can use the icons in
|
|
||||||
# the material Icons class.
|
|
||||||
uses-material-design: true
|
uses-material-design: true
|
||||||
|
|
||||||
# To add assets to your application, add an assets section, like this:
|
# 여기에 에셋 등록
|
||||||
# assets:
|
assets:
|
||||||
# - images/a_dot_burr.jpeg
|
- assets/images/icons8-google-logo-36.png
|
||||||
# - images/a_dot_ham.jpeg
|
- assets/images/icons8-google-logo-48.png
|
||||||
|
- assets/images/icons8-google-logo-48-2.png
|
||||||
# An image asset can refer to one or more resolution-specific "variants", see
|
- assets/images/icons8-google-logo-72.png
|
||||||
# https://flutter.dev/to/resolution-aware-images
|
- assets/images/icons8-google-logo-96.png
|
||||||
|
- assets/images/icons8-google-logo-144.png
|
||||||
# For details regarding adding assets from package dependencies, see
|
- assets/images/icons8-google-logo-192.png
|
||||||
# https://flutter.dev/to/asset-from-package
|
|
||||||
|
|
||||||
# To add custom fonts to your application, add a fonts section here,
|
|
||||||
# in this "flutter" section. Each entry in this list should have a
|
|
||||||
# "family" key with the font family name, and a "fonts" key with a
|
|
||||||
# list giving the asset and other descriptors for the font. For
|
|
||||||
# example:
|
|
||||||
# fonts:
|
|
||||||
# - family: Schyler
|
|
||||||
# fonts:
|
|
||||||
# - asset: fonts/Schyler-Regular.ttf
|
|
||||||
# - asset: fonts/Schyler-Italic.ttf
|
|
||||||
# style: italic
|
|
||||||
# - family: Trajan Pro
|
|
||||||
# fonts:
|
|
||||||
# - asset: fonts/TrajanPro.ttf
|
|
||||||
# - asset: fonts/TrajanPro_Bold.ttf
|
|
||||||
# weight: 700
|
|
||||||
#
|
|
||||||
# For details regarding fonts from package dependencies,
|
|
||||||
# see https://flutter.dev/to/font-from-package
|
|
||||||
|