name: galmoji
description: "ギャル文字"
publish_to: 'none'
version: 1.2.3+11
environment:
sdk: ^3.11.5
dependencies: #flutter pub upgrade --major-versions
flutter:
sdk: flutter
cupertino_icons: ^1.0.8
share_plus: ^13.1.0
shared_preferences: ^2.0.17
google_mobile_ads: ^8.0.0
just_audio: ^0.10.4
flutter_localizations: # flutter gen-l10n
sdk: flutter
intl: ^0.20.2
wakelock_plus: ^1.4.0
in_app_review: ^2.0.11
app_settings: ^7.0.0
dev_dependencies:
flutter_lints: ^6.0.0
flutter_launcher_icons: ^0.14.3 #flutter pub run flutter_launcher_icons
flutter_native_splash: ^2.3.5 #flutter pub run flutter_native_splash:create
flutter_launcher_icons:
android: "launcher_icon"
ios: true
image_path: "assets/icon/icon.png"
adaptive_icon_background: "assets/icon/icon_back.png"
adaptive_icon_foreground: "assets/icon/icon_fore.png"
flutter_native_splash:
color: '#c80064'
image: 'assets/image/splash.png'
color_dark: '#c80064'
image_dark: 'assets/image/splash.png'
fullscreen: true
android_12:
icon_background_color: '#c80064'
image: 'assets/image/splash.png'
icon_background_color_dark: '#c80064'
image_dark: 'assets/image/splash.png'
flutter:
generate: true
uses-material-design: true
config:
enable-swift-package-manager: true
assets:
- assets/image/
- assets/sound/
/// Copyright© ao-system, Inc.
import 'package:flutter/cupertino.dart';
import 'package:google_mobile_ads/google_mobile_ads.dart';
import 'package:galmoji/ad_manager.dart';
class AdBannerWidget extends StatefulWidget {
final AdManager adManager;
const AdBannerWidget({super.key, required this.adManager});
@override
State<AdBannerWidget> createState() => _AdBannerWidgetState();
}
class _AdBannerWidgetState extends State<AdBannerWidget> {
int _lastBannerWidthDp = 0;
bool _isAdLoaded = false;
bool _isLoading = false;
@override
Widget build(BuildContext context) {
return SafeArea(
child: LayoutBuilder(
builder: (context, constraints) {
final int width = constraints.maxWidth.isFinite ? constraints.maxWidth.truncate() : MediaQuery.of(context).size.width.truncate();
final bannerAd = widget.adManager.bannerAd;
if (width > 0) {
WidgetsBinding.instance.addPostFrameCallback((_) {
if (mounted) {
final bannerAd = widget.adManager.bannerAd;
final bool widthChanged = _lastBannerWidthDp != width;
final bool sizeMismatch = bannerAd == null || bannerAd.size.width != width;
if ((widthChanged || !_isAdLoaded || sizeMismatch) && !_isLoading) {
_lastBannerWidthDp = width;
setState(() { _isAdLoaded = false; _isLoading = true; });
widget.adManager.loadAdaptiveBannerAd(width, () {
if (mounted) {
setState(() { _isAdLoaded = true; _isLoading = false; });
}
});
}
}
});
}
if (_isAdLoaded && bannerAd != null) {
return Column(
mainAxisSize: MainAxisSize.min,
children: [
SizedBox(height: 10),
Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
SizedBox(
width: bannerAd.size.width.toDouble(),
height: bannerAd.size.height.toDouble(),
child: AdWidget(ad: bannerAd),
),
],
)
]
);
} else {
return const SizedBox.shrink();
}
},
),
);
}
}
/// Copyright© ao-system, Inc.
import 'dart:async';
import 'dart:io' show Platform;
import 'dart:ui';
import 'package:flutter/foundation.dart' show kIsWeb;
import 'package:google_mobile_ads/google_mobile_ads.dart';
import 'package:flutter/widgets.dart';
import 'package:galmoji/_secrets.dart';
class AdManager {
static String get _adUnitId => Platform.isIOS ? Secrets.adUnitIdIos : Secrets.adUnitIdAndroid;
BannerAd? _bannerAd;
int _lastWidthPx = 0;
VoidCallback? _onLoadedCb;
Timer? _retryTimer;
int _retryAttempt = 0;
BannerAd? get bannerAd => _bannerAd;
/// アプリ起動時の設定
/// UMP(同意管理)を導入したため、手動のNPA設定は不要になった。
static Future<void> initForNPA() async {
if (kIsWeb) {
return;
}
// UMP SDK が保存した同意情報を MobileAds SDK が自動で読み取るため、
// ここで RequestConfiguration を使って NPA を強制する必要はない。
await MobileAds.instance.updateRequestConfiguration(
RequestConfiguration(
tagForChildDirectedTreatment: TagForChildDirectedTreatment.unspecified,
testDeviceIds: Secrets.umpConsentTestDeviceIds, //テストデバイスID:広告の誤クリック防止
),
);
}
Future<void> loadAdaptiveBannerAd(int widthPx, VoidCallback onAdLoaded) async {
if (kIsWeb) {
return;
}
_onLoadedCb = onAdLoaded;
_lastWidthPx = widthPx;
_retryAttempt = 0;
_retryTimer?.cancel();
_retryTimer = null;
_startLoad(widthPx);
}
static AdRequest getAdRequest() {
// ユーザーの同意状態(TCF信号)は、SDKによって自動的に付与される。
// 手動で npa: 1 を送ると、UMPでのユーザーの選択と競合する可能性があるため、空で返す。
// AdRequest(nonPersonalizedAds: true);にはしない
return const AdRequest();
}
Future<void> _startLoad(int widthPx) async {
if (kIsWeb) {
return;
}
_bannerAd?.dispose();
AnchoredAdaptiveBannerAdSize? adaptiveSize;
try {
adaptiveSize = await AdSize.getCurrentOrientationAnchoredAdaptiveBannerAdSize(widthPx);
} catch (_) {
adaptiveSize = null;
}
final AdSize size = adaptiveSize ?? AdSize.fullBanner;
_bannerAd = BannerAd(
adUnitId: _adUnitId,
request: getAdRequest(),
size: size,
listener: BannerAdListener(
onAdLoaded: (ad) {
_retryTimer?.cancel();
_retryTimer = null;
_retryAttempt = 0;
final cb = _onLoadedCb;
if (cb != null) {
cb();
}
},
onAdFailedToLoad: (ad, err) {
ad.dispose();
_scheduleRetry();
},
),
)..load();
}
void _scheduleRetry() {
if (kIsWeb) return;
_retryTimer?.cancel();
_retryTimer = null;
_retryAttempt = (_retryAttempt + 1).clamp(1, 5);
final seconds = _retryAttempt >= 4 ? 30 : (3 << (_retryAttempt - 1));
_retryTimer = Timer(Duration(seconds: seconds), () {
_startLoad(_lastWidthPx > 0 ? _lastWidthPx : 320);
});
}
void dispose() {
_bannerAd?.dispose();
_retryTimer?.cancel();
_retryTimer = null;
}
}
/// Copyright© ao-system, Inc.
import 'dart:async';
import 'package:flutter/foundation.dart' show kIsWeb;
import 'package:google_mobile_ads/google_mobile_ads.dart';
import 'package:flutter/widgets.dart';
import 'package:galmoji/l10n/app_localizations.dart';
import 'package:galmoji/_secrets.dart';
/// UMP状態格納用
class AdUmpState {
final PrivacyOptionsRequirementStatus privacyStatus;
final ConsentStatus consentStatus;
final bool privacyOptionsRequired;
final bool isChecking;
const AdUmpState({
required this.privacyStatus,
required this.consentStatus,
required this.privacyOptionsRequired,
required this.isChecking,
});
AdUmpState copyWith({
PrivacyOptionsRequirementStatus? privacyStatus,
ConsentStatus? consentStatus,
bool? privacyOptionsRequired,
bool? isChecking,
}) {
return AdUmpState(
privacyStatus: privacyStatus ?? this.privacyStatus,
consentStatus: consentStatus ?? this.consentStatus,
privacyOptionsRequired:
privacyOptionsRequired ?? this.privacyOptionsRequired,
isChecking: isChecking ?? this.isChecking,
);
}
static const initial = AdUmpState(
privacyStatus: PrivacyOptionsRequirementStatus.unknown,
consentStatus: ConsentStatus.unknown,
privacyOptionsRequired: false,
isChecking: false,
);
}
//UMPコントローラ
class AdUmpConsentController {
//デバッグ用:同意フォームの表示テスト:EEA地域を強制する(本番ではfalseにすること)
final bool forceEeaForDebug = false;
//デバッグ用:同意フォームの表示テスト:EEA地域を強制するテストデバイスID
static final List<String> _testDeviceIds = Secrets.umpConsentTestDeviceIds;
ConsentRequestParameters _buildParams() {
if (forceEeaForDebug && _testDeviceIds.isNotEmpty) {
return ConsentRequestParameters(
consentDebugSettings: ConsentDebugSettings(
debugGeography: DebugGeography.debugGeographyEea,
testIdentifiers: _testDeviceIds,
),
);
}
return ConsentRequestParameters();
}
//同意情報を更新して状態を返す
Future<AdUmpState> updateConsentInfo({AdUmpState current = AdUmpState.initial}) async {
if (kIsWeb) {
return current;
}
var state = current.copyWith(isChecking: true);
try {
final params = _buildParams();
final completer = Completer<AdUmpState>();
ConsentInformation.instance.requestConsentInfoUpdate(
params,
() async {
//同意フォームが必要なら表示する
ConsentForm.loadAndShowConsentFormIfRequired((formError) async {
final s = await ConsentInformation.instance.getPrivacyOptionsRequirementStatus();
final c = await ConsentInformation.instance.getConsentStatus();
completer.complete(
state.copyWith(
privacyStatus: s,
consentStatus: c,
privacyOptionsRequired: s == PrivacyOptionsRequirementStatus.required,
isChecking: false,
),
);
});
},
(FormError e) {
completer.complete(state.copyWith(isChecking: false));
},
);
return await completer.future;
} catch (_) {
return state.copyWith(isChecking: false);
}
}
//プライバシーオプションフォームを表示
Future<FormError?> showPrivacyOptions() async {
if (kIsWeb) return null;
final completer = Completer<FormError?>();
ConsentForm.showPrivacyOptionsForm((FormError? e) {
completer.complete(e);
});
return completer.future;
}
}
class AdUmpService {
final AdUmpConsentController _adUmpConsentController = AdUmpConsentController();
Future<AdUmpState> updateConsentInfo(AdUmpState current) async {
return await _adUmpConsentController.updateConsentInfo(current: current);
}
Future<void> requestConsentInfoUpdate(ConsentRequestParameters params) async {
final completer = Completer<void>();
ConsentInformation.instance.requestConsentInfoUpdate(
params,
() => completer.complete(),
(FormError error) => completer.completeError(error),
);
return completer.future;
}
Future<FormError?> showPrivacyOptions() async {
return await _adUmpConsentController.showPrivacyOptions();
}
}
extension ConsentStatusL10n on ConsentStatus {
String localized(BuildContext context) {
final l = AppLocalizations.of(context)!;
switch (this) {
case ConsentStatus.obtained:
return l.cmpConsentStatusObtained;
case ConsentStatus.required:
return l.cmpConsentStatusRequired;
case ConsentStatus.notRequired:
return l.cmpConsentStatusNotRequired;
case ConsentStatus.unknown:
return l.cmpConsentStatusUnknown;
}
}
}
/// Copyright© ao-system, Inc.
///
/// @author akira ohmachi
/// @copyright ao-system, Inc.
/// @date 2023-02-06
///
library;
import 'dart:io';
class AdUnitId {
static String get bannerAdUnitId {
if (Platform.isAndroid) {
//return 'ca-app-pub-3940256099942544/6300978111'; //test
return '<YOUR_ANDROID_INTERSTITIAL_AD_UNIT_ID>';
} else if (Platform.isIOS) {
//return 'ca-app-pub-3940256099942544/2934735716'; //test
return '<YOUR_ANDROID_INTERSTITIAL_AD_UNIT_ID>';
} else {
throw UnsupportedError('Unsupported platform');
}
}
static String get interstitialAdUnitId {
if (Platform.isAndroid) {
//return 'ca-app-pub-3940256099942544/1033173712'; //test
return '<YOUR_ANDROID_INTERSTITIAL_AD_UNIT_ID>';
} else if (Platform.isIOS) {
//return 'ca-app-pub-3940256099942544/4411468910'; //test
return '<YOUR_IOS_INTERSTITIAL_AD_UNIT_ID>';
} else {
throw UnsupportedError('Unsupported platform');
}
}
static String get rewardedAdUnitId {
if (Platform.isAndroid) {
//return 'ca-app-pub-3940256099942544/5224354917'; //test
return '<YOUR_ANDROID_REWARDED_AD_UNIT_ID>';
} else if (Platform.isIOS) {
//return 'ca-app-pub-3940256099942544/1712485313'; //test
return '<YOUR_IOS_REWARDED_AD_UNIT_ID>';
} else {
throw UnsupportedError('Unsupported platform');
}
}
}
/// Copyright© ao-system, Inc.
import 'package:flutter/services.dart';
///App Tracking Transparency サービス
///iOS 14以降で広告トラッキングの許可をリクエストする
class AttService {
//チャンネル名をiOS側と一致させる
static const _channel = MethodChannel('aosystem.att');
static final AttService _instance = AttService._internal();
factory AttService() => _instance;
AttService._internal();
///トラッキング許可をリクエスト
///戻り値: AttStatus (enum)
Future<AttStatus> requestTracking() async {
try {
//iOS側からInt値を受け取る
final result = await _channel.invokeMethod<int>('requestTracking');
//enumに変換して返却
return parseAttStatus(result);
} on MissingPluginException catch (_) {
//ハンドラ未登録(iOS以外/旧バイナリ等)はアプリ全体を落とさない
return AttStatus.unknown;
} on PlatformException catch (_) {
return AttStatus.unknown;
}
}
///現在のトラッキング許可状態を取得
///戻り値:AttStatus(enum)
Future<AttStatus> getTrackingStatus() async {
try {
//iOS側からInt値を受け取る
final result = await _channel.invokeMethod<int>('getTrackingStatus');
//enumに変換して返却
return parseAttStatus(result);
} on MissingPluginException catch (_) {
return AttStatus.unknown;
} on PlatformException catch (_) {
return AttStatus.unknown;
}
}
///トラッキングが許可されているか
Future<bool> isTrackingAuthorized() async {
final status = await getTrackingStatus();
return status == AttStatus.authorized;
}
}
//一緒に利用する enum とヘルパー関数
enum AttStatus {
notDetermined, // 0
restricted, // 1
denied, // 2
authorized, // 3
unknown, // 4
}
AttStatus parseAttStatus(int? value) {
switch (value) {
case 0:
return AttStatus.notDetermined;
case 1:
return AttStatus.restricted;
case 2:
return AttStatus.denied;
case 3:
return AttStatus.authorized;
default:
return AttStatus.unknown;
}
}
/// Copyright© ao-system, Inc.
import 'package:just_audio/just_audio.dart';
import 'package:galmoji/const_value.dart';
class AudioPlay {
//音を重ねて連続再生できるようにインスタンスを用意しておき、順繰りに使う。
static final List<AudioPlayer> _player01 = List.generate(6, (_) => AudioPlayer());
int _player01Ptr = 0;
// 最初に音が鳴らない問題を回避するための専用プレイヤー
static final AudioPlayer _warmUpPlayer = AudioPlayer();
double _soundVolume = 0.0;
//constructor
AudioPlay() {
constructor();
}
void constructor() async {
for (int i = 0; i < _player01.length; i++) {
await _player01[i].setVolume(0);
await _player01[i].setAsset(ConstValue.audioHiyokos[i % ConstValue.audioHiyokos.length]);
}
// warm up playerの初期化
await _warmUpPlayer.setAsset(ConstValue.audioZero);
await _warmUpPlayer.load();
playZero();
}
void dispose() {
for (int i = 0; i < _player01.length; i++) {
_player01[i].dispose();
}
_warmUpPlayer.dispose();
}
set soundVolume(double vol) {
_soundVolume = vol;
}
//最初に音が鳴らないのを回避する方法
void playZero() async {
await _warmUpPlayer.play();
await _warmUpPlayer.pause();
await _warmUpPlayer.seek(Duration.zero);
}
//
void play01() async {
if (_soundVolume == 0) {
return;
}
_player01Ptr += 1;
if (_player01Ptr >= _player01.length) {
_player01Ptr = 0;
}
await _player01[_player01Ptr].setVolume(_soundVolume);
await _player01[_player01Ptr].seek(Duration.zero);
await _player01[_player01Ptr].play();
}
}
/// Copyright© ao-system, Inc.
class ConstValue {
//image
static const List<String> imageBackGrounds = [
'assets/image/back001.webp',
'assets/image/back002.webp',
'assets/image/back003.webp',
'assets/image/back004.webp',
'assets/image/back005.webp',
'assets/image/back006.webp',
'assets/image/back007.webp',
'assets/image/back008.webp',
'assets/image/back009.webp',
'assets/image/back010.webp',
'assets/image/back011.webp',
'assets/image/back012.webp',
'assets/image/back013.webp',
'assets/image/back014.webp',
'assets/image/back015.webp',
'assets/image/back016.webp',
'assets/image/back017.webp',
'assets/image/back018.webp',
'assets/image/back019.webp',
'assets/image/back020.webp',
'assets/image/back021.webp',
'assets/image/back022.webp',
'assets/image/back023.webp',
'assets/image/back024.webp',
'assets/image/back025.webp',
'assets/image/back026.webp',
'assets/image/back027.webp',
'assets/image/back028.webp',
'assets/image/back029.webp',
'assets/image/back030.webp',
'assets/image/back031.webp',
'assets/image/back032.webp',
'assets/image/back033.webp',
'assets/image/back034.webp',
'assets/image/back035.webp',
'assets/image/back036.webp',
'assets/image/back037.webp',
'assets/image/back038.webp',
'assets/image/back039.webp',
'assets/image/back040.webp',
'assets/image/back041.webp',
'assets/image/back042.webp',
'assets/image/back043.webp',
'assets/image/back044.webp',
'assets/image/back045.webp',
'assets/image/back046.webp',
'assets/image/back047.webp',
'assets/image/back048.webp',
'assets/image/back049.webp',
'assets/image/back050.webp',
'assets/image/back051.webp',
'assets/image/back052.webp',
'assets/image/back053.webp',
'assets/image/back054.webp',
'assets/image/back055.webp',
'assets/image/back056.webp',
'assets/image/back057.webp',
'assets/image/back058.webp',
'assets/image/back059.webp',
'assets/image/back060.webp',
'assets/image/back061.webp',
'assets/image/back062.webp',
'assets/image/back063.webp',
'assets/image/back064.webp',
'assets/image/back065.webp',
'assets/image/back066.webp',
'assets/image/back067.webp',
'assets/image/back068.webp',
'assets/image/back069.webp',
'assets/image/back070.webp',
'assets/image/back071.webp',
'assets/image/back072.webp',
'assets/image/back073.webp',
'assets/image/back074.webp',
'assets/image/back075.webp',
'assets/image/back076.webp',
'assets/image/back077.webp',
'assets/image/back078.webp',
'assets/image/back079.webp',
'assets/image/back080.webp',
'assets/image/back081.webp',
'assets/image/back082.webp',
'assets/image/back083.webp',
'assets/image/back084.webp',
'assets/image/back085.webp',
'assets/image/back086.webp',
'assets/image/back087.webp',
'assets/image/back088.webp',
'assets/image/back089.webp',
'assets/image/back090.webp',
'assets/image/back091.webp',
'assets/image/back092.webp',
'assets/image/back093.webp',
'assets/image/back094.webp',
'assets/image/back095.webp',
'assets/image/back096.webp',
'assets/image/back097.webp',
'assets/image/back098.webp',
'assets/image/back099.webp',
'assets/image/back100.webp',
];
//sound
static const String audioZero = 'assets/sound/zero.wav'; //無音1秒
static const List<String> audioHiyokos = [
'assets/sound/hiyoko1.wav',
'assets/sound/hiyoko2.wav',
'assets/sound/hiyoko3.wav',
'assets/sound/hiyoko4.wav',
'assets/sound/hiyoko5.wav',
'assets/sound/hiyoko6.wav',
];
//string
static const Map<String,List<String>> galCharKana = {
'現':['王見'],
'科':['禾斗'],
'私':['禾ム'],
'叶':['ロ十'],
'役':['テ殳'],
'例':['イ歹リ'],
'近':['辶斤'],
'健':['イ廴聿'],
'有':['ナ月','囿'],
'名':['タ。'],
'氷':['⊃i⊂','ラ|<'],
'天':['无'],
'本':['夲'],
'来':['來'],
'神':['ネ申'],
'超':['走召'],
'林':['木木'],
'校':['木交'],
'絶':['糸色'],
'級':['糸及'],
'鉄':['金失','金矢'],
'柿':['木市'],
'仔':['イ子'],
'愛してる':['愛ιτゑ','愛Uτゑ'],
'仲良し':['人㊥ょU','イ㊥夜歹ヒ','仲仔'],
'謝':['言身寸'],
'終':['糸冬'],
'対':['文寸'],
'便':['イ更戸斤'],
'桂':['木圭'],
'大':['因'],
'人':['囚'],
'木':['困'],
'巻':['圏'],
'口':['回'],
'幸':['圉'],
'水':['囦'],
'子':['囝'],
'女':['囡'],
'寸':['団'],
'井':['囲'],
'化':['囮'],
'元':['园'],
'爪':['図'],
'不':['囨'],
'古':['固'],
'八土':['囶'],
'玉':['国'],
'令':['囹'],
'四方八方':['4ほぅ圀'],
'書':['圕'],
'員':['圓'],
'日':['Θ','曰'],
'中':['Φ','㊥'],
'チート':['升'],
'行':['テテ','〒〒'],
'給料':['糸合米斗'],
'就活':['京尤シ舌'],
'仕':['イ士'],
'休':['イ木'],
'祝':['ネ兄'],
'読':['言売'],
'欲':['谷欠'],
'願':['原頁'],
'朋':['月月'],
'語':['言吾'],
'認':['言忍'],
'続':['糸売'],
'使':['イ吏'],
'死':['タヒ'],
'上':['㊤'],
'下':['㊦'],
'左':['㊧','ナェ'],
'右':['㊨'],
'印':['㊞','E卩'],
'まじ':['まぢ'],
'てゆうか':['ッ塚'],
'~て言っていて':['~ッ乙ゆウねぇ'],
'あ':['ぁ','ァ','了'],
'い':['ぃ','ィ','レヽ','レ丶','レ)','レ`','L丶','L1','レl'],
'う':['ぅ','ゥ','宀','ヴ','ウ'],
'え':['ぇ','ェ','之','工','ヱ'],
'お':['ぉ','ォ','才'],
'か':['カゝ','カ丶','カヽ','ヵゝ','カ`','カ'],
'が':['ヵゞ'],
'ガ':['カ゛','力゛'],
'き':['(キ','(≠','L≠','‡'],
'く':['<','〈','勹'],
'け':['ヶ','(ナ','レ†','レナ','|ナ','l+','Iナ'],
'こ':['〓','=',']','⊃','⊇'],
'さ':['廾','±','(十','L+','(+','ナ'],
'し':['ι','∪','U'],
'す':['£','フ',],
'せ':['世'],
'そ':['ξ','ζ','`ノ','丶/','ヽ丿'],
'た':['ナ=','+=','†ニ','ナニ','十こ','†こ','ナ⊇','T=','十=','夕'],
'ち':['干','千','于','ろ+'],
'つ':['っ','ッ','⊃','ツ'],
'て':['τ','〒','z','乙'],
'と':['┠','┝','┣','├','`⊂','卜','`c'],
'な':['ナょ','十ょ','†ょ','ナg','†ょゝ','十'],
'に':['(ニ','|=','丨ニ','L=','I=','(⊇','レこ','(二','〓','レニ'],
'ぬ':['йu','ゐ'],
'ね':['йё'],
'の':['/','丿','σ','⊂n','@'],
'は':['\'`','八','l£','(£','ノ|','ノl','レ£','レよ'],
'バ':['ハ〃'],
'パ':['ハo'],
'ひ':['匕'],
'ビ':['ヒ〃'],
'ピ':['ヒo'],
'ふ':['ヴ',',ζ,'],
'ブ':['フ〃'],
'プ':['フo'],
'へ':['~','∧'],
'べ':['ヘ〃'],
'ペ':['ヘo'],
'ほ':['朮','レま'],
'ボ':['ホ〃'],
'ポ':['ホo'],
'ま':['ма','мα'],
'み':['彡','ゐ'],
'む':['£','厶'],
'め':['×','x','χ','乂','〆'],
'も':['м○','мσ','=し','=L'],
'や':['ゃ','ャ'],
'ゆ':['ゅ','ュ'],
'よ':['ょ','ョ','∋','чo','∃'],
'ら':['яа','ζ','b`'],
'り':['L|','l)','レ」','レ)','┗』','└丿','v)','レ」','丶)'],
'る':['ゐ','ゑ','儿','lレ','」レ','|レ','ノレ','/レ'],
'れ':['яё'],
'ろ':['з','З','回'],
'わ':['ゎ','ヮ','wα'],
'を':['щo','ぉ','ε'],
'ん':['ω','冫','w','h','ン','ソ'],
'゙':['〃','”','¨'],
'ー':['→','⇒'],
'。':['o','○'],
};
static const Map<String,List<String>> galCharAlphabet = {
'A':['闩','月','Д','д','@','Å','∀'],
'B':['吕','官','♭','в','ь'],
'C':['匚','匸','¢','⊂','℃'],
'D':['囙'],
'E':['ヨ','巨','臣','巳','ㅌ','ё'],
'F':['孒','下'],
'G':['⊂┐','C┐'],
'H':['丩','н'],
'I':['工','|'],
'J':['し','」','』'],
'K':['|く','κ'],
'L':['└','|'],
'M':['从','ヘ','川','瓜','м'],
'N':['冂','∩','и','И'],
'O':['口','○','ο'],
'P':['尸','ρ'],
'Q':['电','O、'],
'R':['尺','γ','я','Я'],
'S':['丂','$','∫'],
'T':['丁','て','十','т'],
'U':['凵','∪','ц'],
'V':['レ','∨'],
'W':['山','ш','щ','Ш','Щ'],
'X':['乂','χ','×'],
'Y':['ソ','¥','ч','Ч'],
'Z':['乙'],
};
}
/// Copyright© ao-system, Inc.
import 'dart:async';
import 'dart:math';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:share_plus/share_plus.dart';
import 'package:wakelock_plus/wakelock_plus.dart';
import 'package:galmoji/theme_color.dart';
import 'package:galmoji/const_value.dart';
import 'package:galmoji/setting_page.dart';
import 'package:galmoji/ad_banner_widget.dart';
import 'package:galmoji/model.dart';
import 'package:galmoji/audio_play.dart';
import 'package:galmoji/l10n/app_localizations.dart';
import 'package:galmoji/loading_screen.dart';
import 'package:galmoji/main.dart';
class MainHomePage extends StatefulWidget {
const MainHomePage({super.key});
@override
State<MainHomePage> createState() => _MainHomePageState();
}
class _MainHomePageState extends State<MainHomePage> with SingleTickerProviderStateMixin, WidgetsBindingObserver {
late ThemeColor _themeColor;
bool _isReady = false;
late AudioPlay _audioPlay;
bool _areImagesPrecached = false;
final TextEditingController _textEditingController = TextEditingController();
late AnimationController _animationController;
late Animation<double> _opacityAnimation;
int _backImageNumber = 0;
int _lastBackImageNumber = 0;
bool _convertKana = true;
bool _convertAlphabet = false;
late Random _random;
String _inputText = '';
String _convertedText = '';
@override
void initState() {
super.initState();
WidgetsBinding.instance.addObserver(this);
_initState();
}
void _initState() async {
_themeColor = ThemeColor(themeNumber: Model.themeNumber, context: context);
_audioPlay = AudioPlay();
_audioPlay.playZero();
int randomSeed = (DateTime.now()).millisecondsSinceEpoch;
_random = Random(randomSeed);
_wakelock();
// Initialize with a random background image
_backImageNumber = _random.nextInt(ConstValue.imageBackGrounds.length);
_lastBackImageNumber = _backImageNumber;
//background animation
_animationController = AnimationController(
vsync: this,
duration: const Duration(milliseconds: 500),
);
_opacityAnimation = Tween<double>(begin: 0, end: 1).animate(_animationController);
_audioPlay.soundVolume = Model.soundVolume;
if (mounted) {
setState(() {
_isReady = true;
});
}
}
@override
void didChangeDependencies() {
super.didChangeDependencies();
if (!_areImagesPrecached && Model.showBackImage) {
for (var imagePath in ConstValue.imageBackGrounds) {
precacheImage(AssetImage(imagePath), context);
}
_areImagesPrecached = true;
}
}
@override
void dispose() {
WidgetsBinding.instance.removeObserver(this);
WakelockPlus.disable();
_textEditingController.dispose();
_animationController.dispose();
super.dispose();
}
void _wakelock() {
if (Model.wakelockEnabled) {
WakelockPlus.enable();
} else {
WakelockPlus.disable();
}
}
void _backImageChange() {
if (!Model.showBackImage || _animationController.isAnimating) {
return;
}
final int second = (DateTime.now()).millisecondsSinceEpoch ~/ 1000 ~/ 3; //3秒ごとに変化
int newImageNumber = second % ConstValue.imageBackGrounds.length;
if (newImageNumber == _backImageNumber) {
newImageNumber = (newImageNumber + 1) % ConstValue.imageBackGrounds.length;
}
setState(() {
_backImageNumber = newImageNumber;
});
_animationController.forward(from: 0.0).then((_) {
setState(() {
_lastBackImageNumber = _backImageNumber;
});
_animationController.reset();
});
}
void _conversion() {
if (!_isReady) {
return;
}
final text = _inputText;
final buffer = StringBuffer();
// 1. Create a combined map of conversions based on active toggles.
final Map<String, List<String>> conversionMap = {};
if (_convertKana) {
conversionMap.addAll(ConstValue.galCharKana);
}
if (_convertAlphabet) {
conversionMap.addAll(ConstValue.galCharAlphabet);
}
if (conversionMap.isEmpty) {
setState(() {
_convertedText = text;
});
return;
}
// 2. Sort keys by length, descending, to prioritize longer matches.
final sortedKeys = conversionMap.keys.toList()..sort((a, b) => b.length.compareTo(a.length));
// 3. Iterate through the string and convert.
int i = 0;
while (i < text.length) {
String? matchedKey;
for (final key in sortedKeys) {
if (text.startsWith(key, i)) {
matchedKey = key;
break;
}
}
if (matchedKey != null) {
final List<String> values = conversionMap[matchedKey]!;
final String to = values[_random.nextInt(values.length)];
buffer.write(to);
i += matchedKey.length;
} else {
buffer.write(text[i]);
i += 1;
}
}
setState(() {
_convertedText = buffer.toString();
});
}
void _onInputChanged(String text) {
_inputText = text;
_conversion();
}
void _onKanaChanged(bool value) {
_audioPlay.play01();
_backImageChange();
setState(() {
_convertKana = value;
_conversion();
});
}
void _onAlphabetChanged(bool value) {
_audioPlay.play01();
_backImageChange();
setState(() {
_convertAlphabet = value;
_conversion();
});
}
void _onRegenerate() {
_audioPlay.play01();
_conversion();
_backImageChange();
}
void _onCopy(AppLocalizations l) {
_audioPlay.play01();
if (_convertedText.isNotEmpty) {
FocusScope.of(context).unfocus();
Clipboard.setData(ClipboardData(text: _convertedText));
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(l.copied),
duration: Duration(seconds: 2),
),
);
}
}
void _onShare() {
_audioPlay.play01();
if (_convertedText.isNotEmpty) {
SharePlus.instance.share(ShareParams(text: _convertedText));
}
}
Future<void> _openSetting() async {
final updated = await Navigator.push<bool>(
context,
MaterialPageRoute(builder: (_) => const SettingPage()),
);
if (mounted && updated == true) {
MainApp.of(context).rebuildApp();
_themeColor = ThemeColor(themeNumber: Model.themeNumber, context: context);
_audioPlay.soundVolume = Model.soundVolume;
_wakelock();
}
if (mounted) {
setState(() {});
}
}
@override
Widget build(BuildContext context) {
if (!_isReady) {
return const LoadingScreen();
}
final AppLocalizations l = AppLocalizations.of(context)!;
return Container(
decoration: BoxDecoration(
color: _themeColor.mainBackColor,
),
child: Stack(
children: [
if (Model.showBackImage)
Container(
decoration: BoxDecoration(
image: DecorationImage(
image: AssetImage(ConstValue.imageBackGrounds[_lastBackImageNumber]),
fit: BoxFit.cover,
),
),
),
if (Model.showBackImage)
FadeTransition(
opacity: _opacityAnimation,
child: Container(
decoration: BoxDecoration(
image: DecorationImage(
image: AssetImage(ConstValue.imageBackGrounds[_backImageNumber]),
fit: BoxFit.cover,
),
),
),
),
Scaffold(
backgroundColor: Colors.transparent,
appBar: AppBar(
foregroundColor: _themeColor.mainForeColor,
backgroundColor: Colors.transparent,
title: Text(l.appTitle, style: const TextStyle(fontSize: 15.0)),
centerTitle: false,
actions: [
IconButton(
icon: const Icon(Icons.settings),
onPressed: _openSetting,
),
const SizedBox(width: 8),
],
),
body: SafeArea(
child: Column(children:[
Expanded(
child: GestureDetector(
onTap: () => FocusScope.of(context).unfocus(), //背景タップでキーボードを仕舞う
child: SingleChildScrollView(
child: Padding(
padding: const EdgeInsets.all(4.0),
child: Column(children:[
_textFieldInput(l),
const SizedBox(height: 8),
Row(children: [
_toggleKana(l),
const SizedBox(width: 8),
_toggleAlphabet(l),
]),
const SizedBox(height: 8),
_textFieldResult(),
const SizedBox(height: 4),
Row(children: [
_regenerationButton(l),
const SizedBox(width: 8),
_copyClipboardButton(l),
const SizedBox(width: 8),
_shareButton(l),
]),
]),
)
)
)
),
])
),
bottomNavigationBar: AdBannerWidget(adManager: MainApp.of(context).adManager)
)
],
)
);
}
Widget _textFieldInput(AppLocalizations l) {
return Container(
decoration: BoxDecoration(
color: _themeColor.mainButtonBackColor,
borderRadius: BorderRadius.circular(10.0),
),
child: Padding(
padding: const EdgeInsets.all(8.0),
child: SizedBox(
height: 200,
child: TextField(
expands: true,
controller: _textEditingController,
keyboardType: TextInputType.multiline,
maxLines: null,
onChanged: _onInputChanged,
decoration: InputDecoration(
labelText: l.inputText,
border: OutlineInputBorder(),
)
)
)
)
);
}
Widget _toggleKana(AppLocalizations l) {
return Expanded(
child: Container(
decoration: BoxDecoration(
color: _themeColor.mainButtonBackColor,
borderRadius: BorderRadius.circular(10.0),
),
child: Padding(
padding: const EdgeInsets.only(top: 1, left: 8, right: 8, bottom: 1),
child: Row(children: <Widget>[
Expanded(
child: Text(l.convertKana, style: TextStyle(fontSize: 12)),
),
Switch(
value: _convertKana,
onChanged: _onKanaChanged,
),
]),
)
)
);
}
Widget _toggleAlphabet(AppLocalizations l) {
return Expanded(
child: Container(
decoration: BoxDecoration(
color: _themeColor.mainButtonBackColor,
borderRadius: BorderRadius.circular(10.0),
),
child: Padding(
padding: const EdgeInsets.only(top: 1, left: 8, right: 8, bottom: 1),
child: Row(children: <Widget>[
Expanded(
child: Text(l.convertAlphabet, style: TextStyle(fontSize: 12)),
),
Switch(
value: _convertAlphabet,
onChanged: _onAlphabetChanged,
),
]),
)
)
);
}
Widget _textFieldResult() {
return Container(
decoration: BoxDecoration(
color: _themeColor.mainButtonBackColor,
borderRadius: BorderRadius.circular(10.0),
),
width: double.infinity,
child: Padding(
padding: const EdgeInsets.all(12.0),
child: SelectableText(
_convertedText,
style: const TextStyle(fontSize: 16),
minLines: 1,
maxLines: 10,
)
)
);
}
Widget _regenerationButton(AppLocalizations l) {
return Expanded(
child: TextButton(
onPressed: _onRegenerate,
style: TextButton.styleFrom(
backgroundColor: _themeColor.mainButtonBackColor,
textStyle: TextStyle(fontSize: 12),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(10),
),
),
child: Text(l.regenerate),
),
);
}
Widget _copyClipboardButton(AppLocalizations l) {
return Expanded(
child: TextButton(
onPressed: () => _onCopy(l),
style: TextButton.styleFrom(
backgroundColor: _themeColor.mainButtonBackColor,
textStyle: TextStyle(fontSize: 12),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(10),
),
),
child: Text(l.copy),
),
);
}
Widget _shareButton(AppLocalizations l) {
return Expanded(
child: TextButton(
onPressed: _onShare,
style: TextButton.styleFrom(
backgroundColor: _themeColor.mainButtonBackColor,
textStyle: TextStyle(fontSize: 12),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(10),
),
),
child: Text(l.send),
),
);
}
}
/// Copyright© ao-system, Inc.
import 'dart:math';
import 'package:flutter/material.dart';
class LoadingScreen extends StatefulWidget {
const LoadingScreen({super.key});
@override
State<LoadingScreen> createState() => _LoadingScreenState();
}
class _LoadingScreenState extends State<LoadingScreen> with SingleTickerProviderStateMixin {
late AnimationController _animationController;
@override
void initState() {
super.initState();
final randomStart = Random().nextDouble();
_animationController = AnimationController(
vsync: this,
duration: const Duration(seconds: 6),
value: randomStart,
)..repeat();
}
@override
void dispose() {
_animationController.dispose();
super.dispose();
}
Color _rainbowColor(double value) {
final hue = _animationController.value * 360;
return HSVColor.fromAHSV(1, hue, 1, value).toColor();
}
@override
Widget build(BuildContext context) {
final barHeight = MediaQuery.of(context).size.height * 0.4;
return AnimatedBuilder(
animation: _animationController,
builder: (context, _) {
final foreColor = _rainbowColor(1.0);
final backColor = _rainbowColor(0.08);
return Scaffold(
backgroundColor: backColor,
body: Center(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
SizedBox(
height: barHeight,
child: RotatedBox(
quarterTurns: -1,
child: LinearProgressIndicator(
minHeight: 1,
valueColor: AlwaysStoppedAnimation(foreColor),
backgroundColor: Colors.transparent,
),
),
),
const SizedBox(height: 5),
Text(
'LOADING',
style: TextStyle(
color: foreColor,
fontSize: 18,
letterSpacing: 16,
),
),
const SizedBox(height: 5),
SizedBox(
height: barHeight,
child: RotatedBox(
quarterTurns: 1,
child: LinearProgressIndicator(
minHeight: 1,
valueColor: AlwaysStoppedAnimation(foreColor),
backgroundColor: Colors.transparent,
),
),
),
],
),
),
);
},
);
}
}
/// Copyright© ao-system, Inc.
import 'dart:async';
import 'dart:io';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:google_mobile_ads/google_mobile_ads.dart';
import 'package:galmoji/l10n/app_localizations.dart';
import 'package:galmoji/home_page.dart';
import 'package:galmoji/model.dart';
import 'package:galmoji/theme_mode_number.dart';
import 'package:galmoji/parse_locale_tag.dart';
import 'package:galmoji/loading_screen.dart';
import 'package:galmoji/ad_ump_status.dart';
import 'package:galmoji/att_service.dart';
import 'package:galmoji/ad_manager.dart';
void main() async {
WidgetsFlutterBinding.ensureInitialized();
//UI設定
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
SystemChrome.setSystemUIOverlayStyle(
const SystemUiOverlayStyle(
systemNavigationBarColor: Colors.transparent,
statusBarColor: Colors.transparent,
systemNavigationBarContrastEnforced: false,
systemStatusBarContrastEnforced: false,
),
);
runApp(const MainApp());
}
class MainApp extends StatefulWidget {
const MainApp({super.key});
static MainAppState of(BuildContext context) {
return context.findAncestorStateOfType<MainAppState>()!;
}
@override
State<MainApp> createState() => MainAppState();
}
class MainAppState extends State<MainApp> {
late final AdManager adManager;
ThemeMode _themeMode = ThemeMode.system;
Locale? _locale;
bool _hasError = false;
bool _isReady = false;
@override
void initState() {
super.initState();
_initState();
}
void _initState() async {
try {
//ad
adManager = AdManager();
//アプリの基本データ
await Model.ensureReady();
//ATT
//iOSは「アプリがactive/resumed状態」でないとrequestTrackingがダイアログを出さず即座にnotDeterminedを返すため、ライフサイクルがresumedになるまで待つ。
//(iOSは「設定→トラッキング」でトグルを変えるとアプリプロセスをkillして再起動するので、起動時にgetTrackingStatusを読めば常に最新の値が手に入る)
if (!kIsWeb && Platform.isIOS) {
if (await _waitForResumed()) {
final attService = AttService();
//未決定(初回起動)のときだけダイアログ表示。既に決定済みならスキップ。
if (await attService.getTrackingStatus() == AttStatus.notDetermined) {
await attService.requestTracking();
}
}
}
//UMP(ATTの後)
final adUmpConsentController = AdUmpConsentController();
await adUmpConsentController.updateConsentInfo();
//Mobile Ads SDK(同意確定後)
await MobileAds.instance.initialize();
//自前の広告設定
await AdManager.initForNPA();
//UI更新
if (mounted) {
setState(() {
_themeMode = ThemeModeNumber.numberToThemeMode(Model.themeNumber);
_locale = parseLocaleTag(Model.languageCode);
_isReady = true;
});
}
} catch (e) {
if (mounted) {
setState(() {
_hasError = true;
});
}
}
}
@override
void dispose() {
adManager.dispose();
super.dispose();
}
//アプリがactive/resumed状態になるまで待つ。すでにresumedならすぐにtrueを返す。タイムアウト時はfalse。
Future<bool> _waitForResumed({
Duration timeout = const Duration(seconds: 5),
}) async {
final binding = WidgetsBinding.instance;
if (binding.lifecycleState == AppLifecycleState.resumed) {
return true;
}
final completer = Completer<bool>();
late final AppLifecycleListener listener;
listener = AppLifecycleListener(
onStateChange: (state) {
if (state == AppLifecycleState.resumed && !completer.isCompleted) {
completer.complete(true);
}
},
);
try {
return await completer.future.timeout(timeout, onTimeout: () => false);
} finally {
listener.dispose();
}
}
void rebuildApp() {
setState(() {
_themeMode = ThemeModeNumber.numberToThemeMode(Model.themeNumber);
_locale = parseLocaleTag(Model.languageCode);
});
}
ThemeData _createTheme(Brightness brightness, Color seed) {
final colorScheme = ColorScheme.fromSeed(seedColor: seed, brightness: brightness);
return ThemeData(
useMaterial3: true,
colorScheme: colorScheme,
appBarTheme: const AppBarTheme(backgroundColor: Colors.transparent),
elevatedButtonTheme: ElevatedButtonThemeData(
style: ElevatedButton.styleFrom(
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
),
),
sliderTheme: SliderThemeData(
showValueIndicator: ShowValueIndicator.onDrag,
valueIndicatorTextStyle: TextStyle(
color: brightness == Brightness.light ? Colors.white : Colors.black,
),
),
outlinedButtonTheme: OutlinedButtonThemeData(
style: OutlinedButton.styleFrom(
side: BorderSide(color: colorScheme.primary),
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
),
),
);
}
@override
Widget build(BuildContext context) {
if (_hasError) {
return _buildErrorMessage();
}
Color seed = Colors.redAccent;
return MaterialApp(
debugShowCheckedModeBanner: false,
localizationsDelegates: AppLocalizations.localizationsDelegates,
supportedLocales: AppLocalizations.supportedLocales,
locale: _locale,
themeMode: _themeMode,
theme: _createTheme(Brightness.light, seed),
darkTheme: _createTheme(Brightness.dark, seed),
home: _isReady ? const MainHomePage() : const Scaffold(body: LoadingScreen()),
);
}
Widget _buildErrorMessage() {
return const MaterialApp(
debugShowCheckedModeBanner: false,
home: Scaffold(
body: Center(
child: Padding(
padding: EdgeInsets.all(24.0),
child: Text(
'Initialization failed. Please restart the app.',
textAlign: TextAlign.center,
),
),
),
),
);
}
}
/// Copyright© ao-system, Inc.
import 'dart:ui' as ui;
import 'package:shared_preferences/shared_preferences.dart';
import 'package:galmoji/l10n/app_localizations.dart';
class Model {
Model._();
static const String _prefShowBackImage = 'showBackImage';
static const String _prefSoundVolume = 'soundVolume';
static const String _prefWakelockEnabled = 'wakelockEnabled';
static const String _prefThemeNumber = 'themeNumber';
static const String _prefLanguageCode = 'languageCode';
static bool _ready = false;
static bool _showBackImage = true;
static double _soundVolume = 0.3;
static bool _wakelockEnabled = true;
static int _themeNumber = 0;
static String _languageCode = '';
static bool get showBackImage => _showBackImage;
static double get soundVolume => _soundVolume;
static bool get wakelockEnabled => _wakelockEnabled;
static int get themeNumber => _themeNumber;
static String get languageCode => _languageCode;
static Future<void> ensureReady() async {
if (_ready) {
return;
}
final SharedPreferences prefs = await SharedPreferences.getInstance();
//
_showBackImage = prefs.getBool(_prefShowBackImage) ?? true;
_soundVolume = (prefs.getDouble(_prefSoundVolume) ?? 0.3).clamp(0.0, 1.0);
_wakelockEnabled = prefs.getBool(_prefWakelockEnabled) ?? true;
_themeNumber = (prefs.getInt(_prefThemeNumber) ?? 0).clamp(0, 2);
_languageCode = prefs.getString(_prefLanguageCode) ?? ui.PlatformDispatcher.instance.locale.languageCode;
_languageCode = _resolveLanguageCode(_languageCode);
_ready = true;
}
static String _resolveLanguageCode(String code) {
final supported = AppLocalizations.supportedLocales;
if (supported.any((l) => l.languageCode == code)) {
return code;
} else {
return '';
}
}
static Future<void> setShowBackImage(bool value) async {
_showBackImage = value;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setBool(_prefShowBackImage, value);
}
static Future<void> setSoundVolume(double value) async {
_soundVolume = value;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setDouble(_prefSoundVolume, value);
}
static Future<void> setWakelockEnabled(bool value) async {
_wakelockEnabled = value;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setBool(_prefWakelockEnabled, value);
}
static Future<void> setThemeNumber(int value) async {
_themeNumber = value;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setInt(_prefThemeNumber, value);
}
static Future<void> setLanguageCode(String value) async {
_languageCode = value;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setString(_prefLanguageCode, value);
}
}
/// Copyright© ao-system, Inc.
import 'dart:ui';
Locale? parseLocaleTag(String tag) {
if (tag.isEmpty) {
return null;
}
final parts = tag.split('-');
final language = parts[0];
String? script, country;
if (parts.length >= 2) {
parts[1].length == 4 ? script = parts[1] : country = parts[1];
}
if (parts.length >= 3) {
parts[2].length == 4 ? script = parts[2] : country = parts[2];
}
return Locale.fromSubtags(
languageCode: language,
scriptCode: script,
countryCode: country,
);
}
/// Copyright© ao-system, Inc.
///
/// @author akira ohmachi
/// @copyright ao-system, Inc.
/// @date 2023-10-26
///
library;
import 'package:shared_preferences/shared_preferences.dart';
import 'package:galmoji/const_value.dart';
//デバイスに情報を保存
class Preferences {
static bool ready = false;
//この値は常に最新にしておく
static bool _backImageFlag = true;
static double _soundVolume = 0.3;
static bool get backImageFlag {
return _backImageFlag;
}
static double get soundVolume {
return _soundVolume;
}
static Future<void> initial() async {
_backImageFlag = await getBackImageFlag();
_soundVolume = await getSoundVolume();
ready = true;
}
//----------------------------
//背景画像On/Off
static Future<void> setBackImageFlag(bool flag) async {
_backImageFlag = flag;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setBool(ConstValue.prefBackImageFlag, flag);
}
static Future<bool> getBackImageFlag() async {
final SharedPreferences prefs = await SharedPreferences.getInstance();
final bool flag = prefs.getBool(ConstValue.prefBackImageFlag) ?? true;
return flag;
}
//----------------------------
//効果音音量
static Future<void> setSoundVolume(double num) async {
_soundVolume = num;
final SharedPreferences prefs = await SharedPreferences.getInstance();
await prefs.setDouble(ConstValue.prefSoundVolume, num);
}
static Future<double> getSoundVolume() async {
final SharedPreferences prefs = await SharedPreferences.getInstance();
final double num = prefs.getDouble(ConstValue.prefSoundVolume) ?? 0.3;
return num;
}
//----------------------------
}
/// Copyright© ao-system, Inc.
///
/// @author akira ohmachi
/// @copyright ao-system, Inc.
/// @date 2023-11-17
///
library;
import 'package:flutter/material.dart';
import 'package:galmoji/const_value.dart';
import 'package:galmoji/preferences.dart';
import 'package:galmoji/version_state.dart';
import 'package:galmoji/ad_manager.dart';
class SettingPage extends StatefulWidget {
const SettingPage({super.key});
@override
State<SettingPage> createState() => _SettingPageState();
}
class _SettingPageState extends State<SettingPage> {
final AdManager _adManager = AdManager();
bool _isInitialized = false;
//これら変数はUIへの表示や入力の為に一時的に使用される。
bool _backImageFlag = true;
double _soundVolume = 0.0;
//ページ起動時に一度だけ実行される
@override
void initState() {
super.initState();
_initializeAsync();
_adManager.loadBannerAd(() {
setState(() {});
});
}
void _initializeAsync() async {
await Preferences.initial();
_backImageFlag = Preferences.backImageFlag;
_soundVolume = Preferences.soundVolume;
setState(() {
_isInitialized = true;
});
}
//ページ終了時に一度だけ実行される
@override
void dispose() {
_adManager.dispose();
super.dispose();
}
//ページ描画
@override
Widget build(BuildContext context) {
if (!_isInitialized) {
return const Center(
child: CircularProgressIndicator() //初期化中のローディングインジケーターを表示
);
}
return Scaffold(
appBar: AppBar(
centerTitle: true,
elevation: 0,
//設定キャンセルボタン
leading: IconButton(
icon: const Icon(Icons.close),
onPressed: () {
Navigator.of(context).pop(false); //falseを返す
},
),
title: const Text('設定'),
foregroundColor: const Color.fromRGBO(255,255,255,1),
backgroundColor: ConstValue.colorSettingAccent,
actions: [
//設定OKボタン
IconButton(
icon: const Icon(Icons.check),
onPressed: () async {
await Preferences.setBackImageFlag(_backImageFlag);
await Preferences.setSoundVolume(_soundVolume);
if (!mounted) {
return;
}
Navigator.of(context).pop(true); //trueを返す
},
),
],
),
body: SafeArea(
child: Column(children:[
Expanded(
child: GestureDetector(
onTap: () => FocusScope.of(context).unfocus(), //背景タップでキーボードを仕舞う
child: SingleChildScrollView(
child: Padding(
padding: const EdgeInsets.all(20),
child: Column(children: [
Padding(
padding: const EdgeInsets.only(top: 16, left: 16, right: 16, bottom: 16),
child: Row(children:<Widget>[
const Expanded(
child: Text('背景画像表示',style: TextStyle(fontSize: 16)),
),
Switch(
value: _backImageFlag,
onChanged: (bool value) {
setState(() {
_backImageFlag = value;
});
},
activeColor: ConstValue.colorUiActiveColor,
inactiveThumbColor: ConstValue.colorUiInactiveColor,
),
]),
),
_border(),
const Padding(
padding: EdgeInsets.only(top: 18, left: 16, right: 16, bottom: 0),
child: Row(children: [
Text('効果音量',style: TextStyle(fontSize: 16)),
Spacer(),
])
),
Padding(
padding: const EdgeInsets.only(top: 0, left: 16, right: 16, bottom: 6),
child: Row(children: <Widget>[
Text(_soundVolume.toString()),
Expanded(
child: Slider(
value: _soundVolume,
min: 0.0,
max: 1.0,
divisions: 10,
onChanged: (double value) {
setState(() {
_soundVolume = value;
});
},
activeColor: ConstValue.colorUiActiveColor,
inactiveColor: ConstValue.colorUiInactiveColor,
)
)
])
),
_border(),
const Padding(
padding: EdgeInsets.only(top: 24, left: 0, right: 0, bottom: 24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children:[
Text('ギャル文字変換'),
SizedBox(height:15),
Text('日本語文字列を入力欄に入力またはペーストします。ギャル文字への変換結果がリアルタイムに表示されます。'),
SizedBox(height:15),
Text('かなカナ漢字は適宜変換されます。英字は大文字のみです。'),
SizedBox(height:15),
Text('ギャル文字とは、2002年~2005年ぐらいに女子中学生や女子高生の間で流行となった文字遊び。'),
]
),
),
_border(),
Padding(
padding: const EdgeInsets.only(top: 24, left: 0, right: 0, bottom: 36),
child: SizedBox(
child: Text('version ${VersionState.versionLoad()}',
style: const TextStyle(
fontSize: 10,
),
),
),
),
]),
),
),
),
),
//広告
_adManager.widget()
]),
)
);
}
//UIの仕切り用ボーダーライン
Widget _border() {
return Container(
decoration: BoxDecoration(
border: Border(
top: BorderSide(
color: Colors.grey.shade300,
width: 1,
),
),
),
);
}
}
/// Copyright© ao-system, Inc.
import 'package:flutter/material.dart';
import 'package:galmoji/theme_color.dart';
import 'package:galmoji/model.dart';
/// 設定画面専用のカスタムCardウィジェット
class SettingCard extends StatelessWidget {
final Widget child;
final ShapeBorder shape;
final EdgeInsetsGeometry margin;
const SettingCard({
super.key,
required this.child,
this.margin = const EdgeInsets.only(left: 0, top: 12, right: 0, bottom: 0),
}) : shape = const RoundedRectangleBorder(
borderRadius: BorderRadius.all(Radius.circular(12)),
);
const SettingCard.top({
super.key,
required this.child,
this.margin = const EdgeInsets.only(left: 0, top: 12, right: 0, bottom: 0),
}) : shape = const RoundedRectangleBorder(
borderRadius: BorderRadius.only(
topLeft: Radius.circular(12),
topRight: Radius.circular(12),
bottomLeft: Radius.circular(0),
bottomRight: Radius.circular(0),
),
);
const SettingCard.flat({
super.key,
required this.child,
this.margin = const EdgeInsets.only(left: 0, top: 2, right: 0, bottom: 0),
}) : shape = const RoundedRectangleBorder(
borderRadius: BorderRadius.only(
topLeft: Radius.circular(0),
topRight: Radius.circular(0),
bottomLeft: Radius.circular(0),
bottomRight: Radius.circular(0),
),
);
const SettingCard.bottom({
super.key,
required this.child,
this.margin = const EdgeInsets.only(left: 0, top: 2, right: 0, bottom: 0),
}) : shape = const RoundedRectangleBorder(
borderRadius: BorderRadius.only(
topLeft: Radius.circular(0),
topRight: Radius.circular(0),
bottomLeft: Radius.circular(12),
bottomRight: Radius.circular(12),
),
);
@override
Widget build(BuildContext context) {
final themeColor = ThemeColor(
themeNumber: Model.themeNumber,
context: context,
);
return SizedBox(
width: double.infinity,
child: Card(
elevation: 0,
margin: margin,
surfaceTintColor: Colors.transparent,
shadowColor: Colors.transparent,
color: themeColor.cardColor,
shape: shape,
child: child,
),
);
}
}
/// Copyright© ao-system, Inc.
import "dart:async";
import "dart:io";
import "package:app_settings/app_settings.dart";
import "package:flutter/foundation.dart";
import "package:flutter/material.dart";
import 'package:google_mobile_ads/google_mobile_ads.dart';
import 'package:in_app_review/in_app_review.dart';
import "package:galmoji/setting_card.dart";
import 'package:galmoji/theme_color.dart';
import 'package:galmoji/l10n/app_localizations.dart';
import 'package:galmoji/model.dart';
import 'package:galmoji/ad_banner_widget.dart';
import 'package:galmoji/ad_ump_status.dart';
import 'package:galmoji/loading_screen.dart';
import 'package:galmoji/_secrets.dart';
import "package:galmoji/main.dart";
import 'package:galmoji/att_service.dart';
class SettingPage extends StatefulWidget {
const SettingPage({super.key});
@override
State<SettingPage> createState() => _SettingPageState();
}
class _SettingPageState extends State<SettingPage> {
AdUmpState _adUmpState = AdUmpState.initial;
late final AdUmpService _adUmpService;
late ThemeColor _themeColor;
final _inAppReview = InAppReview.instance;
bool _wakelockEnabled = true;
int _themeNumber = 0;
String _languageCode = '';
bool _isReady = false;
//
bool _showBackImage = true;
double _soundVolume = 0.0;
@override
void initState() {
super.initState();
_initState();
}
void _initState() async {
//ump
_adUmpService = AdUmpService();
await _refreshConsentInfo();
//model
_showBackImage = Model.showBackImage;
_soundVolume = Model.soundVolume;
_wakelockEnabled = Model.wakelockEnabled;
_themeNumber = Model.themeNumber;
_languageCode = Model.languageCode;
if (mounted) {
setState(() {
_isReady = true;
});
}
}
@override
void didChangeDependencies() {
super.didChangeDependencies();
_themeColor = ThemeColor(themeNumber: Model.themeNumber, context: context);
}
Future<void> _refreshConsentInfo() async {
final AdUmpState newState = await _adUmpService.updateConsentInfo(_adUmpState);
if (mounted) {
setState(() { _adUmpState = newState; });
}
}
Future<void> _onApply() async {
await Model.setShowBackImage(_showBackImage);
await Model.setSoundVolume(_soundVolume);
await Model.setWakelockEnabled(_wakelockEnabled);
await Model.setThemeNumber(_themeNumber);
await Model.setLanguageCode(_languageCode);
if (!mounted) {
return;
}
Navigator.of(context).pop(true);
}
@override
Widget build(BuildContext context) {
if (!_isReady) {
return LoadingScreen();
}
final l = AppLocalizations.of(context)!;
final TextTheme t = Theme.of(context).textTheme;
return Scaffold(
backgroundColor: _themeColor.backColor,
appBar: AppBar(
backgroundColor: Colors.transparent,
elevation: 0,
leading: IconButton(
icon: const Icon(Icons.close),
onPressed: () => Navigator.of(context).pop(false),
),
title: Text(l.setting),
centerTitle: true,
actions: [
Padding(
padding: const EdgeInsets.only(right: 10),
child:IconButton(
icon: const Icon(Icons.check),
onPressed: _onApply,
)
),
],
),
body: SafeArea(
child: Column(children:[
Expanded(
child: GestureDetector(
onTap: () => FocusScope.of(context).unfocus(), //背景タップでキーボードを仕舞う
child: SingleChildScrollView(
child: Padding(
padding: const EdgeInsets.only(left: 12, right: 12, top: 4, bottom: 100),
child: Column(children: [
_buildShowBackImage(l, t),
_buildSoundVolume(l, t),
_buildWakelockEnabled(l, t),
_buildTheme(l, t),
_buildLanguage(l, t),
_buildReview(l, t),
_buildCmp(l, t),
_buildAtt(l, t),
_buildUsage(l, t),
]),
),
),
),
),
]),
),
bottomNavigationBar: AdBannerWidget(adManager: MainApp.of(context).adManager)
);
}
Widget _buildShowBackImage(AppLocalizations l, TextTheme t) {
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Text(l.showBackImage, style: t.bodyMedium),
trailing: Switch(
value: _showBackImage,
onChanged: (value) {
setState(() {
_showBackImage = value;
});
},
),
),
);
}
Widget _buildSoundVolume(AppLocalizations l, TextTheme t) {
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Text(l.soundVolume, style: t.bodyMedium),
subtitle: Row(
children: [
Text(_soundVolume.toStringAsFixed(1)),
const SizedBox(width: 12),
Expanded(
child: Slider(
value: _soundVolume,
min: 0.0,
max: 1.0,
divisions: 10,
label: _soundVolume.toStringAsFixed(1),
onChanged: (value) {
setState(() {
_soundVolume = double.parse(value.toStringAsFixed(1));
});
},
),
),
],
),
),
);
}
Widget _buildWakelockEnabled(AppLocalizations l, TextTheme t) {
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Text(l.wakelockEnabled, style: t.bodyMedium),
trailing: Switch(
value: _wakelockEnabled,
onChanged: (value) => setState(() => _wakelockEnabled = value),
),
),
);
}
Widget _buildTheme(AppLocalizations l, TextTheme t) {
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
minVerticalPadding: 0,
title: Text(l.theme, style: t.bodyMedium),
trailing: DropdownButton<int>(
value: _themeNumber,
items: [
DropdownMenuItem(value: 0, child: Text(l.systemSetting)),
DropdownMenuItem(value: 1, child: Text(l.lightTheme)),
DropdownMenuItem(value: 2, child: Text(l.darkTheme)),
],
onChanged: (value) {
if (value != null) {
setState(() {
_themeNumber = value;
});
}
},
),
),
);
}
Widget _buildLanguage(AppLocalizations l, TextTheme t) {
final Map<String,String> languageNames = {
'en': 'en: English',
'ja': 'ja: 日本語',
};
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
minVerticalPadding: 0,
title: Text(l.language, style: t.bodyMedium),
trailing: DropdownButton<String?>(
value: _languageCode,
items: [
DropdownMenuItem(value: '', child: Text('Default')),
...languageNames.entries.map((entry) => DropdownMenuItem<String?>(
value: entry.key,
child: Text(entry.value),
)),
],
onChanged: (String? value) {
setState(() {
_languageCode = value ?? '';
});
},
),
),
);
}
Widget _buildReview(AppLocalizations l, TextTheme t) {
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Text(l.reviewApp, style: t.bodyMedium),
subtitle: Column(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
const SizedBox(height: 8),
OutlinedButton.icon(
icon: const Icon(Icons.open_in_new, size: 16),
label: Text(l.reviewStore, style: t.bodySmall),
onPressed: () async {
await _inAppReview.openStoreListing(
appStoreId: Secrets.appStoreId,
);
},
),
],
),
),
);
}
Widget _buildCmp(AppLocalizations l, TextTheme t) {
final showButton = _adUmpState.privacyStatus == PrivacyOptionsRequirementStatus.required;
String statusLabel = l.cmpCheckingRegion;
IconData statusIcon = Icons.help_outline;
switch (_adUmpState.privacyStatus) {
case PrivacyOptionsRequirementStatus.required:
statusLabel = l.cmpRegionRequiresSettings;
statusIcon = Icons.privacy_tip_outlined;
break;
case PrivacyOptionsRequirementStatus.notRequired:
statusLabel = l.cmpRegionNoSettingsRequired;
statusIcon = Icons.check_circle_outline;
break;
case PrivacyOptionsRequirementStatus.unknown:
statusLabel = l.cmpRegionCheckFailed;
statusIcon = Icons.error_outline;
break;
}
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Text(l.cmpSettingsTitle, style: t.bodyMedium),
subtitle: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const SizedBox(height: 8),
Text(l.cmpConsentDescription, style: t.bodySmall),
const SizedBox(height: 16),
Center(
child: Column(
children: [
Chip(
avatar: Icon(statusIcon, size: 18),
label: Text(statusLabel),
),
const SizedBox(height: 6),
Text(
'${l.cmpConsentStatusLabel} ${_adUmpState.consentStatus.localized(context)}',
style: t.bodySmall,
),
if (_adUmpState.consentStatus == ConsentStatus.obtained) ...[
const SizedBox(height: 6),
Text(l.cmpConsentStatusObtainedNote, style: t.bodySmall),
],
if (showButton) ...[
const SizedBox(height: 8),
ElevatedButton.icon(
onPressed: _adUmpState.isChecking
? null
: () async {
try {
await _adUmpService.showPrivacyOptions();
} catch (e) {
//debugPrint('Privacy options error ignored: $e');
}
await _refreshConsentInfo();
},
icon: const Icon(Icons.settings),
label: Text(
_adUmpState.isChecking
? l.cmpConsentStatusChecking
: l.cmpOpenConsentSettings,
),
),
const SizedBox(height: 8),
OutlinedButton.icon(
onPressed: _adUmpState.isChecking ? null : _refreshConsentInfo,
icon: const Icon(Icons.refresh),
label: Text(l.cmpRefreshStatus),
),
const SizedBox(height: 8),
OutlinedButton.icon(
onPressed: () async {
final messenger = ScaffoldMessenger.of(context);
final message = l.cmpResetStatusDone;
await ConsentInformation.instance.reset();
if (!mounted) {
return;
}
setState(() {
_adUmpState = _adUmpState.copyWith(
consentStatus: ConsentStatus.unknown,
);
});
messenger.showSnackBar(SnackBar(content: Text(message)));
},
icon: const Icon(Icons.delete_sweep_outlined),
label: Text(l.cmpResetStatus),
),
],
],
),
),
],
),
),
);
}
Widget _buildAtt(AppLocalizations l, TextTheme t) {
if (kIsWeb || !Platform.isIOS) {
return const SizedBox.shrink();
}
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Text(l.attSettingsTitle, style: t.bodyMedium),
subtitle: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const SizedBox(height: 8),
Text(l.attDescription, style: t.bodySmall),
const SizedBox(height: 8),
FutureBuilder<AttStatus>(
future: AttService().getTrackingStatus(),
builder: (context, snapshot) {
if (snapshot.connectionState == ConnectionState.waiting) {
return Center(
child: Column(
children: [
Chip(
avatar: const Icon(Icons.hourglass_empty),
label: Text(l.attStatusChecking),
),
const SizedBox(height: 8),
OutlinedButton.icon(
onPressed: null,
icon: const Icon(Icons.open_in_new),
label: Text(l.attOpenSettings),
),
],
),
);
}
final status = snapshot.data ?? AttStatus.unknown;
final label = status.name;
return Center(
child: Column(
children: [
Chip(
avatar: const Icon(Icons.track_changes),
label: Text('${l.attStatusLabel} $label'),
),
const SizedBox(height: 8),
OutlinedButton.icon(
onPressed: () => AppSettings.openAppSettings(),
icon: const Icon(Icons.open_in_new, size: 16),
label: Text(l.attOpenSettings, style: t.bodySmall),
),
],
),
);
},
),
],
),
),
);
}
Widget _buildUsage(AppLocalizations l, TextTheme t) {
return SettingCard(
child: ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
title: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(l.usage1, style: t.bodySmall),
const SizedBox(height: 8),
Text(l.usage2, style: t.bodySmall),
const SizedBox(height: 8),
Text(l.usage3, style: t.bodySmall),
const SizedBox(height: 8),
Text(l.usage4, style: t.bodySmall),
],
),
),
);
}
}
/// Copyright© ao-system, Inc.
import 'package:flutter/material.dart';
class ThemeColor {
final int? themeNumber;
final BuildContext context;
ThemeColor({this.themeNumber, required this.context});
Brightness get _effectiveBrightness {
switch (themeNumber) {
case 1:
return Brightness.light;
case 2:
return Brightness.dark;
default:
return Theme.of(context).brightness;
}
}
bool get _isLight => _effectiveBrightness == Brightness.light;
//main page
Color get mainBackColor => _isLight ? Color.fromRGBO(230, 230, 230, 1.0) : Color.fromRGBO(50, 50, 50, 1.0);
Color get mainForeColor => _isLight ? Color.fromRGBO(0, 0, 0, 0.8) : Color.fromRGBO(255, 255, 255, 0.8);
Color get mainButtonBackColor => _isLight ? Color.fromRGBO(255, 255, 255, 0.7) : Color.fromRGBO(0,0,0,0.7);
//setting page
Color get backColor => _isLight ? Colors.grey[200]! : Colors.grey[900]!;
Color get cardColor => _isLight ? Colors.white : Colors.grey[800]!;
Color get appBarForegroundColor => _isLight ? Colors.grey[700]! : Colors.white70;
Color get dropdownColor => cardColor;
Color get borderColor => _isLight ? Colors.grey[300]! : Colors.grey[700]!;
Color get inputFillColor => _isLight ? Colors.grey[50]! : Colors.grey[900]!;
}
/// Copyright© ao-system, Inc.
import 'package:flutter/material.dart';
class ThemeModeNumber {
static ThemeMode numberToThemeMode(int value) {
switch (value) {
case 1:
return ThemeMode.light;
case 2:
return ThemeMode.dark;
default:
return ThemeMode.system;
}
}
}
/// Copyright© ao-system, Inc.
///
/// @author akira ohmachi
/// @copyright ao-system, Inc.
/// @date 2023-01-27
///
library;
class VersionState {
static String _version = '';
//バージョンを記録
static void versionSave(String str) {
_version = str;
}
//バージョンを返す
static String versionLoad() {
return _version;
}
}