Compare commits
13 Commits
Author | SHA1 | Date |
---|---|---|
Kenneth Bruen | f0ccf59db9 | 1 year ago |
Kenneth Bruen | cc7caffffa | 1 year ago |
Kenneth Bruen | 17d8fac893 | 1 year ago |
Kenneth Bruen | 1f48e868b0 | 1 year ago |
Kenneth Bruen | 9d2871405d | 2 years ago |
Kenneth Bruen | 9637551d7a | 2 years ago |
Kenneth Bruen | 2456f7cbda | 2 years ago |
Kenneth Bruen | 0f39a30921 | 2 years ago |
Kenneth Bruen | 1e4ca0c61b | 2 years ago |
Kenneth Bruen | 1269a93624 | 2 years ago |
Kenneth Bruen | a5615fe3cb | 2 years ago |
Kenneth Bruen | 342b870e93 | 2 years ago |
Kenneth Bruen | 0647f260db | 2 years ago |
92 changed files with 6905 additions and 1174 deletions
File diff suppressed because one or more lines are too long
@ -0,0 +1,34 @@ |
|||||||
|
PODS: |
||||||
|
- Flutter (1.0.0) |
||||||
|
- package_info_plus (0.4.5): |
||||||
|
- Flutter |
||||||
|
- shared_preferences_ios (0.0.1): |
||||||
|
- Flutter |
||||||
|
- url_launcher_ios (0.0.1): |
||||||
|
- Flutter |
||||||
|
|
||||||
|
DEPENDENCIES: |
||||||
|
- Flutter (from `Flutter`) |
||||||
|
- package_info_plus (from `.symlinks/plugins/package_info_plus/ios`) |
||||||
|
- shared_preferences_ios (from `.symlinks/plugins/shared_preferences_ios/ios`) |
||||||
|
- url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) |
||||||
|
|
||||||
|
EXTERNAL SOURCES: |
||||||
|
Flutter: |
||||||
|
:path: Flutter |
||||||
|
package_info_plus: |
||||||
|
:path: ".symlinks/plugins/package_info_plus/ios" |
||||||
|
shared_preferences_ios: |
||||||
|
:path: ".symlinks/plugins/shared_preferences_ios/ios" |
||||||
|
url_launcher_ios: |
||||||
|
:path: ".symlinks/plugins/url_launcher_ios/ios" |
||||||
|
|
||||||
|
SPEC CHECKSUMS: |
||||||
|
Flutter: f04841e97a9d0b0a8025694d0796dd46242b2854 |
||||||
|
package_info_plus: 6c92f08e1f853dc01228d6f553146438dafcd14e |
||||||
|
shared_preferences_ios: 548a61f8053b9b8a49ac19c1ffbc8b92c50d68ad |
||||||
|
url_launcher_ios: 839c58cdb4279282219f5e248c3321761ff3c4de |
||||||
|
|
||||||
|
PODFILE CHECKSUM: ef19549a9bc3046e7bb7d2fab4d021637c0c58a3 |
||||||
|
|
||||||
|
COCOAPODS: 1.11.3 |
@ -0,0 +1,39 @@ |
|||||||
|
import 'package:flutter/gestures.dart'; |
||||||
|
import 'package:flutter/rendering.dart'; |
||||||
|
|
||||||
|
TextSpan trainIdSpan({ |
||||||
|
required String rank, |
||||||
|
required String number, |
||||||
|
Locale? locale, |
||||||
|
MouseCursor? mouseCursor, |
||||||
|
void Function(PointerEnterEvent)? onEnter, |
||||||
|
void Function(PointerExitEvent)? onExit, |
||||||
|
GestureRecognizer? recognizer, |
||||||
|
String? semanticsLabel, |
||||||
|
bool? spellOut, |
||||||
|
TextStyle? style, |
||||||
|
}) => TextSpan( |
||||||
|
children: [ |
||||||
|
TextSpan( |
||||||
|
text: rank, |
||||||
|
style: TextStyle( |
||||||
|
inherit: true, |
||||||
|
color: rank.startsWith('IC') |
||||||
|
? const Color.fromARGB(255, 0, 255, 0) |
||||||
|
: rank.startsWith('IR') |
||||||
|
? const Color.fromARGB(255, 255, 0, 0) |
||||||
|
: null, |
||||||
|
), |
||||||
|
), |
||||||
|
const TextSpan(text: ' '), |
||||||
|
TextSpan(text: number), |
||||||
|
], |
||||||
|
locale: locale, |
||||||
|
mouseCursor: mouseCursor, |
||||||
|
onEnter: onEnter, |
||||||
|
onExit: onExit, |
||||||
|
recognizer: recognizer, |
||||||
|
semanticsLabel: semanticsLabel, |
||||||
|
spellOut: spellOut, |
||||||
|
style: style, |
||||||
|
); |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,217 @@ |
|||||||
|
// GENERATED CODE - DO NOT MODIFY BY HAND |
||||||
|
|
||||||
|
part of 'train_data.dart'; |
||||||
|
|
||||||
|
// ************************************************************************** |
||||||
|
// JsonSerializableGenerator |
||||||
|
// ************************************************************************** |
||||||
|
|
||||||
|
_$_TrainData _$$_TrainDataFromJson(Map<String, dynamic> json) => _$_TrainData( |
||||||
|
rank: json['rank'] as String, |
||||||
|
number: json['number'] as String, |
||||||
|
date: json['date'] as String, |
||||||
|
operator: json['operator'] as String, |
||||||
|
groups: (json['groups'] as List<dynamic>) |
||||||
|
.map((e) => TrainDataGroup.fromJson(e as Map<String, dynamic>)) |
||||||
|
.toList(), |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataToJson(_$_TrainData instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'rank': instance.rank, |
||||||
|
'number': instance.number, |
||||||
|
'date': instance.date, |
||||||
|
'operator': instance.operator, |
||||||
|
'groups': instance.groups, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataGroup _$$_TrainDataGroupFromJson(Map<String, dynamic> json) => |
||||||
|
_$_TrainDataGroup( |
||||||
|
route: TrainDataRoute.fromJson(json['route'] as Map<String, dynamic>), |
||||||
|
stations: (json['stations'] as List<dynamic>) |
||||||
|
.map((e) => TrainDataStation.fromJson(e as Map<String, dynamic>)) |
||||||
|
.toList(), |
||||||
|
status: json['status'] == null |
||||||
|
? null |
||||||
|
: TrainDataStatus.fromJson(json['status'] as Map<String, dynamic>), |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataGroupToJson(_$_TrainDataGroup instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'route': instance.route, |
||||||
|
'stations': instance.stations, |
||||||
|
'status': instance.status, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataRoute _$$_TrainDataRouteFromJson(Map<String, dynamic> json) => |
||||||
|
_$_TrainDataRoute( |
||||||
|
from: json['from'] as String, |
||||||
|
to: json['to'] as String, |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataRouteToJson(_$_TrainDataRoute instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'from': instance.from, |
||||||
|
'to': instance.to, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataStation _$$_TrainDataStationFromJson(Map<String, dynamic> json) => |
||||||
|
_$_TrainDataStation( |
||||||
|
name: json['name'] as String, |
||||||
|
linkName: json['linkName'] as String, |
||||||
|
km: json['km'] as int, |
||||||
|
stoppingTime: json['stoppingTime'] as int?, |
||||||
|
platform: json['platform'] as String?, |
||||||
|
arrival: json['arrival'] == null |
||||||
|
? null |
||||||
|
: StationArrDepTime.fromJson(json['arrival'] as Map<String, dynamic>), |
||||||
|
departure: json['departure'] == null |
||||||
|
? null |
||||||
|
: StationArrDepTime.fromJson( |
||||||
|
json['departure'] as Map<String, dynamic>), |
||||||
|
notes: (json['notes'] as List<dynamic>) |
||||||
|
.map((e) => const TrainDataNoteConverter() |
||||||
|
.fromJson(e as Map<String, dynamic>)) |
||||||
|
.toList(), |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataStationToJson(_$_TrainDataStation instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'name': instance.name, |
||||||
|
'linkName': instance.linkName, |
||||||
|
'km': instance.km, |
||||||
|
'stoppingTime': instance.stoppingTime, |
||||||
|
'platform': instance.platform, |
||||||
|
'arrival': instance.arrival, |
||||||
|
'departure': instance.departure, |
||||||
|
'notes': |
||||||
|
instance.notes.map(const TrainDataNoteConverter().toJson).toList(), |
||||||
|
}; |
||||||
|
|
||||||
|
_$_StationArrDepTime _$$_StationArrDepTimeFromJson(Map<String, dynamic> json) => |
||||||
|
_$_StationArrDepTime( |
||||||
|
scheduleTime: DateTime.parse(json['scheduleTime'] as String), |
||||||
|
status: json['status'] == null |
||||||
|
? null |
||||||
|
: StationArrDepTimeStatus.fromJson( |
||||||
|
json['status'] as Map<String, dynamic>), |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_StationArrDepTimeToJson( |
||||||
|
_$_StationArrDepTime instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'scheduleTime': instance.scheduleTime.toIso8601String(), |
||||||
|
'status': instance.status, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_StationArrDepTimeStatus _$$_StationArrDepTimeStatusFromJson( |
||||||
|
Map<String, dynamic> json) => |
||||||
|
_$_StationArrDepTimeStatus( |
||||||
|
delay: json['delay'] as int, |
||||||
|
real: json['real'] as bool, |
||||||
|
cancelled: json['cancelled'] as bool, |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_StationArrDepTimeStatusToJson( |
||||||
|
_$_StationArrDepTimeStatus instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'delay': instance.delay, |
||||||
|
'real': instance.real, |
||||||
|
'cancelled': instance.cancelled, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataStatus _$$_TrainDataStatusFromJson(Map<String, dynamic> json) => |
||||||
|
_$_TrainDataStatus( |
||||||
|
delay: json['delay'] as int, |
||||||
|
station: json['station'] as String, |
||||||
|
state: $enumDecode(_$TrainDataStatusStateEnumMap, json['state']), |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataStatusToJson(_$_TrainDataStatus instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'delay': instance.delay, |
||||||
|
'station': instance.station, |
||||||
|
'state': _$TrainDataStatusStateEnumMap[instance.state]!, |
||||||
|
}; |
||||||
|
|
||||||
|
const _$TrainDataStatusStateEnumMap = { |
||||||
|
TrainDataStatusState.passing: 'passing', |
||||||
|
TrainDataStatusState.arrival: 'arrival', |
||||||
|
TrainDataStatusState.departure: 'departure', |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataNoteTrainNumberChange _$$_TrainDataNoteTrainNumberChangeFromJson( |
||||||
|
Map<String, dynamic> json) => |
||||||
|
_$_TrainDataNoteTrainNumberChange( |
||||||
|
kind: json['kind'] as String? ?? "trainNumberChange", |
||||||
|
rank: json['rank'] as String, |
||||||
|
number: json['number'] as String, |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataNoteTrainNumberChangeToJson( |
||||||
|
_$_TrainDataNoteTrainNumberChange instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'kind': instance.kind, |
||||||
|
'rank': instance.rank, |
||||||
|
'number': instance.number, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataNoteDepartsAs _$$_TrainDataNoteDepartsAsFromJson( |
||||||
|
Map<String, dynamic> json) => |
||||||
|
_$_TrainDataNoteDepartsAs( |
||||||
|
kind: json['kind'] as String? ?? "departsAs", |
||||||
|
rank: json['rank'] as String, |
||||||
|
number: json['number'] as String, |
||||||
|
departureDate: DateTime.parse(json['departureDate'] as String), |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataNoteDepartsAsToJson( |
||||||
|
_$_TrainDataNoteDepartsAs instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'kind': instance.kind, |
||||||
|
'rank': instance.rank, |
||||||
|
'number': instance.number, |
||||||
|
'departureDate': instance.departureDate.toIso8601String(), |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataNoteDetachingWagons _$$_TrainDataNoteDetachingWagonsFromJson( |
||||||
|
Map<String, dynamic> json) => |
||||||
|
_$_TrainDataNoteDetachingWagons( |
||||||
|
kind: json['kind'] as String? ?? "detachingWagons", |
||||||
|
station: json['station'] as String, |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataNoteDetachingWagonsToJson( |
||||||
|
_$_TrainDataNoteDetachingWagons instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'kind': instance.kind, |
||||||
|
'station': instance.station, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataNoteReceivingWagons _$$_TrainDataNoteReceivingWagonsFromJson( |
||||||
|
Map<String, dynamic> json) => |
||||||
|
_$_TrainDataNoteReceivingWagons( |
||||||
|
kind: json['kind'] as String? ?? "receivingWagons", |
||||||
|
station: json['station'] as String, |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataNoteReceivingWagonsToJson( |
||||||
|
_$_TrainDataNoteReceivingWagons instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'kind': instance.kind, |
||||||
|
'station': instance.station, |
||||||
|
}; |
||||||
|
|
||||||
|
_$_TrainDataNoteUnknown _$$_TrainDataNoteUnknownFromJson( |
||||||
|
Map<String, dynamic> json) => |
||||||
|
_$_TrainDataNoteUnknown( |
||||||
|
kind: json['kind'] as String, |
||||||
|
extra: json['extra'] as Map<String, dynamic>, |
||||||
|
); |
||||||
|
|
||||||
|
Map<String, dynamic> _$$_TrainDataNoteUnknownToJson( |
||||||
|
_$_TrainDataNoteUnknown instance) => |
||||||
|
<String, dynamic>{ |
||||||
|
'kind': instance.kind, |
||||||
|
'extra': instance.extra, |
||||||
|
}; |
@ -0,0 +1,94 @@ |
|||||||
|
import 'package:timezone/timezone.dart' as tz; |
||||||
|
|
||||||
|
enum UiTimeZoneType { |
||||||
|
ro, |
||||||
|
local, |
||||||
|
utc, |
||||||
|
iana, |
||||||
|
} |
||||||
|
|
||||||
|
extension UITimeZoneTypeName on UiTimeZoneType { |
||||||
|
String get userInterfaceName => (const { |
||||||
|
UiTimeZoneType.iana: 'Fus orar IANA', |
||||||
|
UiTimeZoneType.local: 'Local', |
||||||
|
UiTimeZoneType.ro: 'România', |
||||||
|
UiTimeZoneType.utc: 'UTC', |
||||||
|
})[this]!; |
||||||
|
} |
||||||
|
|
||||||
|
const Map<UiTimeZoneType, UiTimeZone Function(String)> fromSerStringConstructors = { |
||||||
|
UiTimeZoneType.ro: RoUiTimeZone.fromSerString, |
||||||
|
UiTimeZoneType.local: LocalUiTimeZone.fromSerString, |
||||||
|
UiTimeZoneType.utc: UtcUiTimeZone.fromSerString, |
||||||
|
UiTimeZoneType.iana: IanaUiTimeZone.fromSerString, |
||||||
|
}; |
||||||
|
|
||||||
|
abstract class UiTimeZone { |
||||||
|
final UiTimeZoneType type; |
||||||
|
|
||||||
|
const UiTimeZone({required this.type}); |
||||||
|
|
||||||
|
DateTime convertDateTime(DateTime dt); |
||||||
|
|
||||||
|
factory UiTimeZone.fromSerString(String ser) { |
||||||
|
final arr = ser.split('\n'); |
||||||
|
return fromSerStringConstructors.map((key, value) => MapEntry(key.name, value))[arr[0]]!(ser); |
||||||
|
} |
||||||
|
|
||||||
|
String toSerString() { |
||||||
|
return '${type.name}\n'; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
class RoUiTimeZone extends UiTimeZone { |
||||||
|
static final roTz = tz.getLocation('Europe/Bucharest'); |
||||||
|
|
||||||
|
const RoUiTimeZone() : super(type: UiTimeZoneType.ro); |
||||||
|
|
||||||
|
factory RoUiTimeZone.fromSerString(String ser) => const RoUiTimeZone(); |
||||||
|
|
||||||
|
@override |
||||||
|
DateTime convertDateTime(DateTime dt) { |
||||||
|
return tz.TZDateTime.from(dt, roTz); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
class LocalUiTimeZone extends UiTimeZone { |
||||||
|
const LocalUiTimeZone() : super(type: UiTimeZoneType.local); |
||||||
|
|
||||||
|
factory LocalUiTimeZone.fromSerString(String ser) => LocalUiTimeZone(); |
||||||
|
|
||||||
|
@override |
||||||
|
DateTime convertDateTime(DateTime dt) => dt.toLocal(); |
||||||
|
} |
||||||
|
|
||||||
|
class UtcUiTimeZone extends UiTimeZone { |
||||||
|
const UtcUiTimeZone() : super(type: UiTimeZoneType.utc); |
||||||
|
|
||||||
|
factory UtcUiTimeZone.fromSerString(String ser) => UtcUiTimeZone(); |
||||||
|
|
||||||
|
@override |
||||||
|
DateTime convertDateTime(DateTime dt) => dt.toUtc(); |
||||||
|
} |
||||||
|
|
||||||
|
class IanaUiTimeZone extends UiTimeZone { |
||||||
|
late final tz.Location location; |
||||||
|
|
||||||
|
IanaUiTimeZone({required String ianaName}): super(type: UiTimeZoneType.iana) { |
||||||
|
location = tz.getLocation(ianaName); |
||||||
|
} |
||||||
|
|
||||||
|
factory IanaUiTimeZone.fromSerString(String ser) => IanaUiTimeZone( |
||||||
|
ianaName: ser.split('\n').skip(1).join('\n'), |
||||||
|
); |
||||||
|
|
||||||
|
@override |
||||||
|
DateTime convertDateTime(DateTime dt) { |
||||||
|
return tz.TZDateTime.from(dt, location); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
String toSerString() { |
||||||
|
return '${type.name}\n${location.name}'; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,198 @@ |
|||||||
|
import 'package:fluent_ui/fluent_ui.dart'; |
||||||
|
import 'package:flutter/services.dart'; |
||||||
|
import 'package:info_tren/pages/about/about_page.dart'; |
||||||
|
import 'package:url_launcher/url_launcher.dart'; |
||||||
|
|
||||||
|
class AboutPageFluent extends AboutPageShared { |
||||||
|
const AboutPageFluent({super.key}); |
||||||
|
|
||||||
|
@override |
||||||
|
State<AboutPageShared> createState() => AboutPageStateFluent(); |
||||||
|
} |
||||||
|
|
||||||
|
class AboutPageStateFluent extends AboutPageState<AboutPageFluent> { |
||||||
|
@override |
||||||
|
Widget build(BuildContext context) { |
||||||
|
return NavigationView( |
||||||
|
appBar: NavigationAppBar( |
||||||
|
title: Text(pageTitle), |
||||||
|
), |
||||||
|
content: SingleChildScrollView( |
||||||
|
child: Column( |
||||||
|
mainAxisSize: MainAxisSize.min, |
||||||
|
crossAxisAlignment: CrossAxisAlignment.stretch, |
||||||
|
children: [ |
||||||
|
Center( |
||||||
|
child: Text( |
||||||
|
'Info Tren', |
||||||
|
style: FluentTheme.of(context).typography.display, |
||||||
|
), |
||||||
|
), |
||||||
|
if (packageInfo != null) |
||||||
|
Center( |
||||||
|
child: Text( |
||||||
|
packageInfo!.packageName, |
||||||
|
style: FluentTheme.of(context).typography.caption, |
||||||
|
), |
||||||
|
), |
||||||
|
// ListTile( |
||||||
|
// title: Text(versionTitleText), |
||||||
|
// subtitle: localChangelog.isEmpty ? null : Text(localChangelog.first.title), |
||||||
|
// ), |
||||||
|
const Divider(), |
||||||
|
for (final log in mergedChangelogs) ...[ |
||||||
|
Padding( |
||||||
|
padding: const EdgeInsets.fromLTRB(8, 8, 8, 0), |
||||||
|
child: Row( |
||||||
|
crossAxisAlignment: CrossAxisAlignment.center, |
||||||
|
children: [ |
||||||
|
Expanded( |
||||||
|
child: Text( |
||||||
|
log.version.toString(), |
||||||
|
style: FluentTheme.of(context).typography.title, |
||||||
|
), |
||||||
|
), |
||||||
|
if (localChangelog.isNotEmpty && log.version == localChangelog.first.version) |
||||||
|
Container( |
||||||
|
decoration: BoxDecoration( |
||||||
|
border: Border.all( |
||||||
|
color: FluentTheme.of(context).inactiveColor, |
||||||
|
width: 1, |
||||||
|
), |
||||||
|
borderRadius: BorderRadius.circular(20), |
||||||
|
), |
||||||
|
child: Padding( |
||||||
|
padding: const EdgeInsets.all(4), |
||||||
|
child: Text( |
||||||
|
currentVersionText, |
||||||
|
style: const TextStyle( |
||||||
|
inherit: true, |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
if (remoteChangelog.isNotEmpty && log.version == remoteChangelog.first.version && (localChangelog.isEmpty || localChangelog.first.version != log.version)) |
||||||
|
Container( |
||||||
|
decoration: BoxDecoration( |
||||||
|
border: Border.all( |
||||||
|
color: Colors.green, |
||||||
|
width: 1, |
||||||
|
), |
||||||
|
borderRadius: BorderRadius.circular(20), |
||||||
|
), |
||||||
|
child: Padding( |
||||||
|
padding: const EdgeInsets.all(4), |
||||||
|
child: Text( |
||||||
|
latestVersionText, |
||||||
|
style: TextStyle( |
||||||
|
inherit: true, |
||||||
|
color: Colors.green, |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
if (AboutPageState.download == 'apk' && log.apkLink != null) |
||||||
|
GestureDetector( |
||||||
|
onSecondaryTap: () { |
||||||
|
Clipboard.setData(ClipboardData(text: log.apkLink!.toString())); |
||||||
|
// ScaffoldMessenger.of(context).showSnackBar(const SnackBar( |
||||||
|
// content: Text('Link copied to clipboard'), |
||||||
|
// )); |
||||||
|
}, |
||||||
|
onLongPress: () { |
||||||
|
Clipboard.setData(ClipboardData(text: log.apkLink!.toString())); |
||||||
|
// ScaffoldMessenger.of(context).showSnackBar(const SnackBar( |
||||||
|
// content: Text('Link copied to clipboard'), |
||||||
|
// )); |
||||||
|
}, |
||||||
|
onTap: () { |
||||||
|
launchUrl( |
||||||
|
log.apkLink!, |
||||||
|
mode: LaunchMode.externalApplication, |
||||||
|
); |
||||||
|
}, |
||||||
|
behavior: HitTestBehavior.translucent, |
||||||
|
child: const Tooltip( |
||||||
|
message: 'Download APK', |
||||||
|
child: Padding( |
||||||
|
padding: EdgeInsets.all(4), |
||||||
|
child: Icon(FluentIcons.download), |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
if (AboutPageState.download == 'linux' && log.linuxLink != null) |
||||||
|
GestureDetector( |
||||||
|
onSecondaryTap: () { |
||||||
|
Clipboard.setData(ClipboardData(text: log.linuxLink!.toString())); |
||||||
|
// ScaffoldMessenger.of(context).showSnackBar(const SnackBar( |
||||||
|
// content: Text('Link copied to clipboard'), |
||||||
|
// )); |
||||||
|
}, |
||||||
|
onLongPress: () { |
||||||
|
Clipboard.setData(ClipboardData(text: log.linuxLink!.toString())); |
||||||
|
// ScaffoldMessenger.of(context).showSnackBar(const SnackBar( |
||||||
|
// content: Text('Link copied to clipboard'), |
||||||
|
// )); |
||||||
|
}, |
||||||
|
onTap: () { |
||||||
|
launchUrl( |
||||||
|
log.linuxLink!, |
||||||
|
mode: LaunchMode.externalApplication, |
||||||
|
); |
||||||
|
}, |
||||||
|
behavior: HitTestBehavior.translucent, |
||||||
|
child: const Tooltip( |
||||||
|
message: 'Download Linux ZIP', |
||||||
|
child: Padding( |
||||||
|
padding: EdgeInsets.all(4), |
||||||
|
child: Icon(FluentIcons.download), |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
if (AboutPageState.download == 'windows' && log.windowsLink != null) |
||||||
|
GestureDetector( |
||||||
|
onSecondaryTap: () { |
||||||
|
Clipboard.setData(ClipboardData(text: log.windowsLink!.toString())); |
||||||
|
// ScaffoldMessenger.of(context).showSnackBar(const SnackBar( |
||||||
|
// content: Text('Link copied to clipboard'), |
||||||
|
// )); |
||||||
|
}, |
||||||
|
onLongPress: () { |
||||||
|
Clipboard.setData(ClipboardData(text: log.windowsLink!.toString())); |
||||||
|
// ScaffoldMessenger.of(context).showSnackBar(const SnackBar( |
||||||
|
// content: Text('Link copied to clipboard'), |
||||||
|
// )); |
||||||
|
}, |
||||||
|
onTap: () { |
||||||
|
launchUrl( |
||||||
|
log.windowsLink!, |
||||||
|
mode: LaunchMode.externalApplication, |
||||||
|
); |
||||||
|
}, |
||||||
|
behavior: HitTestBehavior.translucent, |
||||||
|
child: const Tooltip( |
||||||
|
message: 'Download Windows App ZIP', |
||||||
|
child: Padding( |
||||||
|
padding: EdgeInsets.all(4), |
||||||
|
child: Icon(FluentIcons.download), |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
], |
||||||
|
), |
||||||
|
), |
||||||
|
Padding( |
||||||
|
padding: const EdgeInsets.all(8.0), |
||||||
|
child: RichText( |
||||||
|
text: TextSpan( |
||||||
|
text: log.description, |
||||||
|
), |
||||||
|
), |
||||||
|
), |
||||||
|
], |
||||||
|
], |
||||||
|
), |
||||||
|
), |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,38 @@ |
|||||||
|
import 'package:flutter/widgets.dart'; |
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart'; |
||||||
|
import 'package:info_tren/models.dart'; |
||||||
|
import 'package:info_tren/pages/settings/settings_page_cupertino.dart'; |
||||||
|
import 'package:info_tren/pages/settings/settings_page_fluent.dart'; |
||||||
|
import 'package:info_tren/pages/settings/settings_page_material.dart'; |
||||||
|
import 'package:info_tren/providers.dart'; |
||||||
|
|
||||||
|
class SettingsPage extends ConsumerWidget { |
||||||
|
const SettingsPage({super.key,}); |
||||||
|
|
||||||
|
static const String routeName = '/settings'; |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build(BuildContext context, WidgetRef ref) { |
||||||
|
final uiDesign = ref.watch(uiDesignProvider); |
||||||
|
|
||||||
|
switch (uiDesign) { |
||||||
|
case UiDesign.MATERIAL: |
||||||
|
return const SettingsPageMaterial(); |
||||||
|
case UiDesign.CUPERTINO: |
||||||
|
return const SettingsPageCupertino(); |
||||||
|
case UiDesign.FLUENT: |
||||||
|
return const SettingsPageFluent(); |
||||||
|
default: |
||||||
|
throw UnmatchedUiDesignException(uiDesign); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
abstract class SettingsPageShared extends StatelessWidget { |
||||||
|
final String pageTitle = 'Setări'; |
||||||
|
final String appearanceTitle = 'Aspect'; |
||||||
|
final String timeZoneTitle = 'Fus orar'; |
||||||
|
|
||||||
|
const SettingsPageShared({super.key}); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,96 @@ |
|||||||
|
import 'package:flutter/cupertino.dart'; |
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart'; |
||||||
|
import 'package:info_tren/models.dart'; |
||||||
|
import 'package:info_tren/pages/settings/setings_page.dart'; |
||||||
|
import 'package:info_tren/providers.dart'; |
||||||
|
|
||||||
|
class SettingsPageCupertino extends SettingsPageShared { |
||||||
|
const SettingsPageCupertino({super.key,}); |
||||||
|
|
||||||
|
Future<T?> singleChoice<T>({required BuildContext context, required List<T> choices, required String Function(T) labelBuilder, String? title}) async { |
||||||
|
return await showCupertinoModalPopup<T>( |
||||||
|
context: context, |
||||||
|
builder: (context) { |
||||||
|
return CupertinoActionSheet( |
||||||
|
title: title != null ? Text(title) : null, |
||||||
|
actions: choices.map((c) => CupertinoActionSheetAction( |
||||||
|
onPressed: () { |
||||||
|
Navigator.of(context).pop(c); |
||||||
|
}, |
||||||
|
child: Text(labelBuilder(c)), |
||||||
|
)).toList(), |
||||||
|
cancelButton: CupertinoActionSheetAction( |
||||||
|
child: const Text('Anulare'), |
||||||
|
onPressed: () { |
||||||
|
Navigator.of(context).pop(); |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
}, |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build(BuildContext context) { |
||||||
|
return CupertinoPageScaffold( |
||||||
|
navigationBar: CupertinoNavigationBar( |
||||||
|
previousPageTitle: 'Info Tren', |
||||||
|
middle: Text(pageTitle), |
||||||
|
), |
||||||
|
child: Builder( |
||||||
|
builder: (context) { |
||||||
|
final mq = MediaQuery.of(context); |
||||||
|
return SingleChildScrollView( |
||||||
|
child: Column( |
||||||
|
children: [ |
||||||
|
SizedBox( |
||||||
|
height: mq.padding.top, |
||||||
|
), |
||||||
|
Consumer( |
||||||
|
builder: (context, ref, _) { |
||||||
|
final currentUiDesign = ref.watch(uiDesignProvider); |
||||||
|
return CupertinoListTile( |
||||||
|
title: Text(appearanceTitle), |
||||||
|
trailing: Text(currentUiDesign.userInterfaceName), |
||||||
|
onTap: () async { |
||||||
|
final choice = await singleChoice( |
||||||
|
context: context, |
||||||
|
choices: UiDesign.values, |
||||||
|
labelBuilder: (UiDesign ud) => ud.userInterfaceName, |
||||||
|
title: appearanceTitle, |
||||||
|
); |
||||||
|
if (choice != null) { |
||||||
|
ref.read(uiDesignProvider.notifier).set(choice); |
||||||
|
} |
||||||
|
}, |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
Consumer( |
||||||
|
builder: (context, ref, _) { |
||||||
|
final currentTZ = ref.watch(uiTimeZoneProvider); |
||||||
|
return CupertinoListTile( |
||||||
|
title: Text(timeZoneTitle), |
||||||
|
trailing: Text(currentTZ.type.userInterfaceName), |
||||||
|
onTap: () async { |
||||||
|
final choice = await singleChoice( |
||||||
|
context: context, |
||||||
|
choices: UiTimeZoneType.values.where((tz) => tz != UiTimeZoneType.iana).toList(), |
||||||
|
labelBuilder: (UiTimeZoneType utzt) => utzt.userInterfaceName, |
||||||
|
title: timeZoneTitle, |
||||||
|
); |
||||||
|
if (choice != null) { |
||||||
|
ref.read(uiTimeZoneProvider.notifier).set(UiTimeZone.fromSerString('${choice.name}\n')); |
||||||
|
} |
||||||
|
}, |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
], |
||||||
|
), |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,65 @@ |
|||||||
|
import 'package:fluent_ui/fluent_ui.dart'; |
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart'; |
||||||
|
import 'package:info_tren/models.dart'; |
||||||
|
import 'package:info_tren/pages/settings/setings_page.dart'; |
||||||
|
import 'package:info_tren/providers.dart'; |
||||||
|
|
||||||
|
class SettingsPageFluent extends SettingsPageShared { |
||||||
|
const SettingsPageFluent({super.key,}); |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build(BuildContext context) { |
||||||
|
return NavigationView( |
||||||
|
appBar: NavigationAppBar( |
||||||
|
title: Text(pageTitle), |
||||||
|
), |
||||||
|
content: SingleChildScrollView( |
||||||
|
child: Column( |
||||||
|
children: [ |
||||||
|
Consumer( |
||||||
|
builder: (context, ref, _) { |
||||||
|
final currentUiDesign = ref.watch(uiDesignProvider); |
||||||
|
return ListTile( |
||||||
|
title: Text(appearanceTitle), |
||||||
|
trailing: ComboBox<UiDesign>( |
||||||
|
items: UiDesign.values.map((d) => ComboBoxItem( |
||||||
|
value: d, |
||||||
|
child: Text(d.userInterfaceName), |
||||||
|
)).toList(), |
||||||
|
value: currentUiDesign, |
||||||
|
onChanged: (newUiDesign) { |
||||||
|
ref.read(uiDesignProvider.notifier).set(newUiDesign); |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
Consumer( |
||||||
|
builder: (context, ref, _) { |
||||||
|
final currentTZ = ref.watch(uiTimeZoneProvider); |
||||||
|
return ListTile( |
||||||
|
title: Text(timeZoneTitle), |
||||||
|
trailing: ComboBox<UiTimeZoneType>( |
||||||
|
items: UiTimeZoneType.values.where((tz) => tz != UiTimeZoneType.iana).map((tzt) => ComboBoxItem( |
||||||
|
value: tzt, |
||||||
|
child: Text(tzt.userInterfaceName), |
||||||
|
)).toList(), |
||||||
|
value: currentTZ.type, |
||||||
|
onChanged: (newTZ) { |
||||||
|
if (newTZ != null) { |
||||||
|
ref.read(uiTimeZoneProvider.notifier).set(UiTimeZone.fromSerString('${newTZ.name}\n')); |
||||||
|
} |
||||||
|
else { |
||||||
|
ref.read(uiTimeZoneProvider.notifier).set(null); |
||||||
|
} |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
], |
||||||
|
), |
||||||
|
), |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,66 @@ |
|||||||
|
import 'package:flutter/material.dart'; |
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart'; |
||||||
|
import 'package:info_tren/models.dart'; |
||||||
|
import 'package:info_tren/pages/settings/setings_page.dart'; |
||||||
|
import 'package:info_tren/providers.dart'; |
||||||
|
|
||||||
|
class SettingsPageMaterial extends SettingsPageShared { |
||||||
|
const SettingsPageMaterial({super.key,}); |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build(BuildContext context) { |
||||||
|
return Scaffold( |
||||||
|
appBar: AppBar( |
||||||
|
title: Text(pageTitle), |
||||||
|
centerTitle: true, |
||||||
|
), |
||||||
|
body: SingleChildScrollView( |
||||||
|
child: Column( |
||||||
|
children: [ |
||||||
|
Consumer( |
||||||
|
builder: (context, ref, _) { |
||||||
|
final currentUiDesign = ref.watch(uiDesignProvider); |
||||||
|
return ListTile( |
||||||
|
title: Text(appearanceTitle), |
||||||
|
trailing: DropdownButton<UiDesign>( |
||||||
|
items: UiDesign.values.map((d) => DropdownMenuItem( |
||||||
|
value: d, |
||||||
|
child: Text(d.userInterfaceName), |
||||||
|
)).toList(), |
||||||
|
value: currentUiDesign, |
||||||
|
onChanged: (newUiDesign) { |
||||||
|
ref.read(uiDesignProvider.notifier).set(newUiDesign); |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
Consumer( |
||||||
|
builder: (context, ref, _) { |
||||||
|
final currentTZ = ref.watch(uiTimeZoneProvider); |
||||||
|
return ListTile( |
||||||
|
title: Text(timeZoneTitle), |
||||||
|
trailing: DropdownButton<UiTimeZoneType>( |
||||||
|
items: UiTimeZoneType.values.where((tz) => tz != UiTimeZoneType.iana).map((tzt) => DropdownMenuItem( |
||||||
|
value: tzt, |
||||||
|
child: Text(tzt.userInterfaceName), |
||||||
|
)).toList(), |
||||||
|
value: currentTZ.type, |
||||||
|
onChanged: (newTZ) { |
||||||
|
if (newTZ != null) { |
||||||
|
ref.read(uiTimeZoneProvider.notifier).set(UiTimeZone.fromSerString('${newTZ.name}\n')); |
||||||
|
} |
||||||
|
else { |
||||||
|
ref.read(uiTimeZoneProvider.notifier).set(null); |
||||||
|
} |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
}, |
||||||
|
), |
||||||
|
], |
||||||
|
), |
||||||
|
), |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,120 @@ |
|||||||
|
// GENERATED CODE - DO NOT MODIFY BY HAND |
||||||
|
|
||||||
|
part of 'providers.dart'; |
||||||
|
|
||||||
|
// ************************************************************************** |
||||||
|
// RiverpodGenerator |
||||||
|
// ************************************************************************** |
||||||
|
|
||||||
|
String _$trainInfoHash() => r'd25aabc3ba656acf6497ec6831e11892178b22c9'; |
||||||
|
|
||||||
|
/// Copied from Dart SDK |
||||||
|
class _SystemHash { |
||||||
|
_SystemHash._(); |
||||||
|
|
||||||
|
static int combine(int hash, int value) { |
||||||
|
// ignore: parameter_assignments |
||||||
|
hash = 0x1fffffff & (hash + value); |
||||||
|
// ignore: parameter_assignments |
||||||
|
hash = 0x1fffffff & (hash + ((0x0007ffff & hash) << 10)); |
||||||
|
return hash ^ (hash >> 6); |
||||||
|
} |
||||||
|
|
||||||
|
static int finish(int hash) { |
||||||
|
// ignore: parameter_assignments |
||||||
|
hash = 0x1fffffff & (hash + ((0x03ffffff & hash) << 3)); |
||||||
|
// ignore: parameter_assignments |
||||||
|
hash = hash ^ (hash >> 11); |
||||||
|
return 0x1fffffff & (hash + ((0x00003fff & hash) << 15)); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
typedef TrainInfoRef = FutureProviderRef<TrainData>; |
||||||
|
|
||||||
|
/// See also [trainInfo]. |
||||||
|
@ProviderFor(trainInfo) |
||||||
|
const trainInfoProvider = TrainInfoFamily(); |
||||||
|
|
||||||
|
/// See also [trainInfo]. |
||||||
|
class TrainInfoFamily extends Family<AsyncValue<TrainData>> { |
||||||
|
/// See also [trainInfo]. |
||||||
|
const TrainInfoFamily(); |
||||||
|
|
||||||
|
/// See also [trainInfo]. |
||||||
|
TrainInfoProvider call({ |
||||||
|
required String trainNumber, |
||||||
|
DateTime? date, |
||||||
|
}) { |
||||||
|
return TrainInfoProvider( |
||||||
|
trainNumber: trainNumber, |
||||||
|
date: date, |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
TrainInfoProvider getProviderOverride( |
||||||
|
covariant TrainInfoProvider provider, |
||||||
|
) { |
||||||
|
return call( |
||||||
|
trainNumber: provider.trainNumber, |
||||||
|
date: provider.date, |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
static const Iterable<ProviderOrFamily>? _dependencies = null; |
||||||
|
|
||||||
|
@override |
||||||
|
Iterable<ProviderOrFamily>? get dependencies => _dependencies; |
||||||
|
|
||||||
|
static const Iterable<ProviderOrFamily>? _allTransitiveDependencies = null; |
||||||
|
|
||||||
|
@override |
||||||
|
Iterable<ProviderOrFamily>? get allTransitiveDependencies => |
||||||
|
_allTransitiveDependencies; |
||||||
|
|
||||||
|
@override |
||||||
|
String? get name => r'trainInfoProvider'; |
||||||
|
} |
||||||
|
|
||||||
|
/// See also [trainInfo]. |
||||||
|
class TrainInfoProvider extends FutureProvider<TrainData> { |
||||||
|
/// See also [trainInfo]. |
||||||
|
TrainInfoProvider({ |
||||||
|
required this.trainNumber, |
||||||
|
this.date, |
||||||
|
}) : super.internal( |
||||||
|
(ref) => trainInfo( |
||||||
|
ref, |
||||||
|
trainNumber: trainNumber, |
||||||
|
date: date, |
||||||
|
), |
||||||
|
from: trainInfoProvider, |
||||||
|
name: r'trainInfoProvider', |
||||||
|
debugGetCreateSourceHash: |
||||||
|
const bool.fromEnvironment('dart.vm.product') |
||||||
|
? null |
||||||
|
: _$trainInfoHash, |
||||||
|
dependencies: TrainInfoFamily._dependencies, |
||||||
|
allTransitiveDependencies: TrainInfoFamily._allTransitiveDependencies, |
||||||
|
); |
||||||
|
|
||||||
|
final String trainNumber; |
||||||
|
final DateTime? date; |
||||||
|
|
||||||
|
@override |
||||||
|
bool operator ==(Object other) { |
||||||
|
return other is TrainInfoProvider && |
||||||
|
other.trainNumber == trainNumber && |
||||||
|
other.date == date; |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
int get hashCode { |
||||||
|
var hash = _SystemHash.combine(0, runtimeType.hashCode); |
||||||
|
hash = _SystemHash.combine(hash, trainNumber.hashCode); |
||||||
|
hash = _SystemHash.combine(hash, date.hashCode); |
||||||
|
|
||||||
|
return _SystemHash.finish(hash); |
||||||
|
} |
||||||
|
} |
||||||
|
// ignore_for_file: unnecessary_raw_strings, subtype_of_sealed_class, invalid_use_of_internal_member, do_not_use_environment, prefer_const_constructors, public_member_api_docs, avoid_private_typedef_functions |
@ -1,12 +1,7 @@ |
|||||||
import 'package:info_tren/models.dart'; |
import 'package:info_tren/models.dart'; |
||||||
|
|
||||||
String stateToString(TrainDataState state) { |
String stateToString(TrainDataState state) => switch (state) { |
||||||
switch(state) { |
TrainDataState.passing => 'trecere fără oprire', |
||||||
case TrainDataState.PASSING: |
TrainDataState.arrival => 'sosire', |
||||||
return 'trecere fără oprire'; |
TrainDataState.departure => 'plecare', |
||||||
case TrainDataState.ARRIVAL: |
}; |
||||||
return 'sosire'; |
|
||||||
case TrainDataState.DEPARTURE: |
|
||||||
return 'plecare'; |
|
||||||
} |
|
||||||
} |
|
||||||
|
@ -0,0 +1,8 @@ |
|||||||
|
[Desktop Entry] |
||||||
|
Type=Application |
||||||
|
Encoding=UTF-8 |
||||||
|
Name=Info Tren |
||||||
|
Comment= |
||||||
|
Exec= |
||||||
|
Icon= |
||||||
|
Terminal=False |
@ -0,0 +1,39 @@ |
|||||||
|
#! /bin/sh |
||||||
|
install_dir="$1" |
||||||
|
if [ -z "$install_dir" ]; then |
||||||
|
echo "Please specify a directory to install InfoTren to." |
||||||
|
echo "Example:" |
||||||
|
echo " $0 ~/infotren" |
||||||
|
exit 3 |
||||||
|
fi |
||||||
|
|
||||||
|
if [ ! -d "$install_dir" ]; then |
||||||
|
if [ -d $(dirname "$install_dir") ]; then |
||||||
|
mkdir "$install_dir" |
||||||
|
else |
||||||
|
echo "$install_dir doesn't exist. Please specify a directory to install InfoTren to." |
||||||
|
echo "Example:" |
||||||
|
echo " $0 ~/infotren" |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
fi |
||||||
|
|
||||||
|
if [ ! -f ./infotren.desktop ]; then |
||||||
|
if [ -f "$(dirname $0)/infotren.desktop" ]; then |
||||||
|
cd "$(dirname $0)" |
||||||
|
else |
||||||
|
echo "Run this script from inside the infotren directory." |
||||||
|
exit 2 |
||||||
|
fi |
||||||
|
fi |
||||||
|
|
||||||
|
echo "Installing InfoTren to $install_dir" |
||||||
|
cp -r . "$install_dir" |
||||||
|
if [ -z "$XDG_DATA_HOME" ]; then |
||||||
|
XDG_DATA_HOME=~/.local/share |
||||||
|
fi |
||||||
|
if [ ! -d "$XDG_DATA_HOME/applications" ]; then |
||||||
|
mkdir -p "$XDG_DATA_HOME/applications" |
||||||
|
fi |
||||||
|
echo "Installing infotren.desktop to $XDG_DATA_HOME/applications/infotren.desktop" |
||||||
|
cat infotren.desktop | sed "s|Exec=|Exec=$install_dir/info_tren|g" > "$XDG_DATA_HOME/applications/infotren.desktop" |
@ -1 +1,2 @@ |
|||||||
|
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig" |
||||||
#include "ephemeral/Flutter-Generated.xcconfig" |
#include "ephemeral/Flutter-Generated.xcconfig" |
||||||
|
@ -1 +1,2 @@ |
|||||||
|
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig" |
||||||
#include "ephemeral/Flutter-Generated.xcconfig" |
#include "ephemeral/Flutter-Generated.xcconfig" |
||||||
|
@ -0,0 +1,40 @@ |
|||||||
|
platform :osx, '10.14' |
||||||
|
|
||||||
|
# CocoaPods analytics sends network stats synchronously affecting flutter build latency. |
||||||
|
ENV['COCOAPODS_DISABLE_STATS'] = 'true' |
||||||
|
|
||||||
|
project 'Runner', { |
||||||
|
'Debug' => :debug, |
||||||
|
'Profile' => :release, |
||||||
|
'Release' => :release, |
||||||
|
} |
||||||
|
|
||||||
|
def flutter_root |
||||||
|
generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'ephemeral', 'Flutter-Generated.xcconfig'), __FILE__) |
||||||
|
unless File.exist?(generated_xcode_build_settings_path) |
||||||
|
raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure \"flutter pub get\" is executed first" |
||||||
|
end |
||||||
|
|
||||||
|
File.foreach(generated_xcode_build_settings_path) do |line| |
||||||
|
matches = line.match(/FLUTTER_ROOT\=(.*)/) |
||||||
|
return matches[1].strip if matches |
||||||
|
end |
||||||
|
raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Flutter-Generated.xcconfig, then run \"flutter pub get\"" |
||||||
|
end |
||||||
|
|
||||||
|
require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root) |
||||||
|
|
||||||
|
flutter_macos_podfile_setup |
||||||
|
|
||||||
|
target 'Runner' do |
||||||
|
use_frameworks! |
||||||
|
use_modular_headers! |
||||||
|
|
||||||
|
flutter_install_all_macos_pods File.dirname(File.realpath(__FILE__)) |
||||||
|
end |
||||||
|
|
||||||
|
post_install do |installer| |
||||||
|
installer.pods_project.targets.each do |target| |
||||||
|
flutter_additional_macos_build_settings(target) |
||||||
|
end |
||||||
|
end |
@ -0,0 +1,41 @@ |
|||||||
|
PODS: |
||||||
|
- dynamic_color (0.0.2): |
||||||
|
- FlutterMacOS |
||||||
|
- FlutterMacOS (1.0.0) |
||||||
|
- package_info_plus (0.0.1): |
||||||
|
- FlutterMacOS |
||||||
|
- shared_preferences_foundation (0.0.1): |
||||||
|
- Flutter |
||||||
|
- FlutterMacOS |
||||||
|
- url_launcher_macos (0.0.1): |
||||||
|
- FlutterMacOS |
||||||
|
|
||||||
|
DEPENDENCIES: |
||||||
|
- dynamic_color (from `Flutter/ephemeral/.symlinks/plugins/dynamic_color/macos`) |
||||||
|
- FlutterMacOS (from `Flutter/ephemeral`) |
||||||
|
- package_info_plus (from `Flutter/ephemeral/.symlinks/plugins/package_info_plus/macos`) |
||||||
|
- shared_preferences_foundation (from `Flutter/ephemeral/.symlinks/plugins/shared_preferences_foundation/darwin`) |
||||||
|
- url_launcher_macos (from `Flutter/ephemeral/.symlinks/plugins/url_launcher_macos/macos`) |
||||||
|
|
||||||
|
EXTERNAL SOURCES: |
||||||
|
dynamic_color: |
||||||
|
:path: Flutter/ephemeral/.symlinks/plugins/dynamic_color/macos |
||||||
|
FlutterMacOS: |
||||||
|
:path: Flutter/ephemeral |
||||||
|
package_info_plus: |
||||||
|
:path: Flutter/ephemeral/.symlinks/plugins/package_info_plus/macos |
||||||
|
shared_preferences_foundation: |
||||||
|
:path: Flutter/ephemeral/.symlinks/plugins/shared_preferences_foundation/darwin |
||||||
|
url_launcher_macos: |
||||||
|
:path: Flutter/ephemeral/.symlinks/plugins/url_launcher_macos/macos |
||||||
|
|
||||||
|
SPEC CHECKSUMS: |
||||||
|
dynamic_color: 2eaa27267de1ca20d879fbd6e01259773fb1670f |
||||||
|
FlutterMacOS: 8f6f14fa908a6fb3fba0cd85dbd81ec4b251fb24 |
||||||
|
package_info_plus: 02d7a575e80f194102bef286361c6c326e4c29ce |
||||||
|
shared_preferences_foundation: 5b919d13b803cadd15ed2dc053125c68730e5126 |
||||||
|
url_launcher_macos: d2691c7dd33ed713bf3544850a623080ec693d95 |
||||||
|
|
||||||
|
PODFILE CHECKSUM: 353c8bcc5d5b0994e508d035b5431cfe18c1dea7 |
||||||
|
|
||||||
|
COCOAPODS: 1.12.1 |
@ -0,0 +1,17 @@ |
|||||||
|
flutter/ephemeral/ |
||||||
|
|
||||||
|
# Visual Studio user-specific files. |
||||||
|
*.suo |
||||||
|
*.user |
||||||
|
*.userosscache |
||||||
|
*.sln.docstates |
||||||
|
|
||||||
|
# Visual Studio build-related files. |
||||||
|
x64/ |
||||||
|
x86/ |
||||||
|
|
||||||
|
# Visual Studio cache files |
||||||
|
# files ending in .cache can be ignored |
||||||
|
*.[Cc]ache |
||||||
|
# but keep track of directories ending in .cache |
||||||
|
!*.[Cc]ache/ |
@ -0,0 +1,101 @@ |
|||||||
|
# Project-level configuration. |
||||||
|
cmake_minimum_required(VERSION 3.14) |
||||||
|
project(info_tren LANGUAGES CXX) |
||||||
|
|
||||||
|
# The name of the executable created for the application. Change this to change |
||||||
|
# the on-disk name of your application. |
||||||
|
set(BINARY_NAME "info_tren") |
||||||
|
|
||||||
|
# Explicitly opt in to modern CMake behaviors to avoid warnings with recent |
||||||
|
# versions of CMake. |
||||||
|
cmake_policy(SET CMP0063 NEW) |
||||||
|
|
||||||
|
# Define build configuration option. |
||||||
|
get_property(IS_MULTICONFIG GLOBAL PROPERTY GENERATOR_IS_MULTI_CONFIG) |
||||||
|
if(IS_MULTICONFIG) |
||||||
|
set(CMAKE_CONFIGURATION_TYPES "Debug;Profile;Release" |
||||||
|
CACHE STRING "" FORCE) |
||||||
|
else() |
||||||
|
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) |
||||||
|
set(CMAKE_BUILD_TYPE "Debug" CACHE |
||||||
|
STRING "Flutter build mode" FORCE) |
||||||
|
set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS |
||||||
|
"Debug" "Profile" "Release") |
||||||
|
endif() |
||||||
|
endif() |
||||||
|
# Define settings for the Profile build mode. |
||||||
|
set(CMAKE_EXE_LINKER_FLAGS_PROFILE "${CMAKE_EXE_LINKER_FLAGS_RELEASE}") |
||||||
|
set(CMAKE_SHARED_LINKER_FLAGS_PROFILE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE}") |
||||||
|
set(CMAKE_C_FLAGS_PROFILE "${CMAKE_C_FLAGS_RELEASE}") |
||||||
|
set(CMAKE_CXX_FLAGS_PROFILE "${CMAKE_CXX_FLAGS_RELEASE}") |
||||||
|
|
||||||
|
# Use Unicode for all projects. |
||||||
|
add_definitions(-DUNICODE -D_UNICODE) |
||||||
|
|
||||||
|
# Compilation settings that should be applied to most targets. |
||||||
|
# |
||||||
|
# Be cautious about adding new options here, as plugins use this function by |
||||||
|
# default. In most cases, you should add new options to specific targets instead |
||||||
|
# of modifying this function. |
||||||
|
function(APPLY_STANDARD_SETTINGS TARGET) |
||||||
|
target_compile_features(${TARGET} PUBLIC cxx_std_17) |
||||||
|
target_compile_options(${TARGET} PRIVATE /W4 /WX /wd"4100") |
||||||
|
target_compile_options(${TARGET} PRIVATE /EHsc) |
||||||
|
target_compile_definitions(${TARGET} PRIVATE "_HAS_EXCEPTIONS=0") |
||||||
|
target_compile_definitions(${TARGET} PRIVATE "$<$<CONFIG:Debug>:_DEBUG>") |
||||||
|
endfunction() |
||||||
|
|
||||||
|
# Flutter library and tool build rules. |
||||||
|
set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") |
||||||
|
add_subdirectory(${FLUTTER_MANAGED_DIR}) |
||||||
|
|
||||||
|
# Application build; see runner/CMakeLists.txt. |
||||||
|
add_subdirectory("runner") |
||||||
|
|
||||||
|
# Generated plugin build rules, which manage building the plugins and adding |
||||||
|
# them to the application. |
||||||
|
include(flutter/generated_plugins.cmake) |
||||||
|
|
||||||
|
|
||||||
|
# === Installation === |
||||||
|
# Support files are copied into place next to the executable, so that it can |
||||||
|
# run in place. This is done instead of making a separate bundle (as on Linux) |
||||||
|
# so that building and running from within Visual Studio will work. |
||||||
|
set(BUILD_BUNDLE_DIR "$<TARGET_FILE_DIR:${BINARY_NAME}>") |
||||||
|
# Make the "install" step default, as it's required to run. |
||||||
|
set(CMAKE_VS_INCLUDE_INSTALL_TO_DEFAULT_BUILD 1) |
||||||
|
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) |
||||||
|
set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) |
||||||
|
endif() |
||||||
|
|
||||||
|
set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") |
||||||
|
set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}") |
||||||
|
|
||||||
|
install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" |
||||||
|
COMPONENT Runtime) |
||||||
|
|
||||||
|
install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||||
|
COMPONENT Runtime) |
||||||
|
|
||||||
|
install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||||
|
COMPONENT Runtime) |
||||||
|
|
||||||
|
if(PLUGIN_BUNDLED_LIBRARIES) |
||||||
|
install(FILES "${PLUGIN_BUNDLED_LIBRARIES}" |
||||||
|
DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||||
|
COMPONENT Runtime) |
||||||
|
endif() |
||||||
|
|
||||||
|
# Fully re-copy the assets directory on each build to avoid having stale files |
||||||
|
# from a previous install. |
||||||
|
set(FLUTTER_ASSET_DIR_NAME "flutter_assets") |
||||||
|
install(CODE " |
||||||
|
file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") |
||||||
|
" COMPONENT Runtime) |
||||||
|
install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" |
||||||
|
DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) |
||||||
|
|
||||||
|
# Install the AOT library on non-Debug builds only. |
||||||
|
install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||||
|
CONFIGURATIONS Profile;Release |
||||||
|
COMPONENT Runtime) |
@ -0,0 +1,104 @@ |
|||||||
|
# This file controls Flutter-level build steps. It should not be edited. |
||||||
|
cmake_minimum_required(VERSION 3.14) |
||||||
|
|
||||||
|
set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") |
||||||
|
|
||||||
|
# Configuration provided via flutter tool. |
||||||
|
include(${EPHEMERAL_DIR}/generated_config.cmake) |
||||||
|
|
||||||
|
# TODO: Move the rest of this into files in ephemeral. See |
||||||
|
# https://github.com/flutter/flutter/issues/57146. |
||||||
|
set(WRAPPER_ROOT "${EPHEMERAL_DIR}/cpp_client_wrapper") |
||||||
|
|
||||||
|
# === Flutter Library === |
||||||
|
set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/flutter_windows.dll") |
||||||
|
|
||||||
|
# Published to parent scope for install step. |
||||||
|
set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) |
||||||
|
set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) |
||||||
|
set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) |
||||||
|
set(AOT_LIBRARY "${PROJECT_DIR}/build/windows/app.so" PARENT_SCOPE) |
||||||
|
|
||||||
|
list(APPEND FLUTTER_LIBRARY_HEADERS |
||||||
|
"flutter_export.h" |
||||||
|
"flutter_windows.h" |
||||||
|
"flutter_messenger.h" |
||||||
|
"flutter_plugin_registrar.h" |
||||||
|
"flutter_texture_registrar.h" |
||||||
|
) |
||||||
|
list(TRANSFORM FLUTTER_LIBRARY_HEADERS PREPEND "${EPHEMERAL_DIR}/") |
||||||
|
add_library(flutter INTERFACE) |
||||||
|
target_include_directories(flutter INTERFACE |
||||||
|
"${EPHEMERAL_DIR}" |
||||||
|
) |
||||||
|
target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}.lib") |
||||||
|
add_dependencies(flutter flutter_assemble) |
||||||
|
|
||||||
|
# === Wrapper === |
||||||
|
list(APPEND CPP_WRAPPER_SOURCES_CORE |
||||||
|
"core_implementations.cc" |
||||||
|
"standard_codec.cc" |
||||||
|
) |
||||||
|
list(TRANSFORM CPP_WRAPPER_SOURCES_CORE PREPEND "${WRAPPER_ROOT}/") |
||||||
|
list(APPEND CPP_WRAPPER_SOURCES_PLUGIN |
||||||
|
"plugin_registrar.cc" |
||||||
|
) |
||||||
|
list(TRANSFORM CPP_WRAPPER_SOURCES_PLUGIN PREPEND "${WRAPPER_ROOT}/") |
||||||
|
list(APPEND CPP_WRAPPER_SOURCES_APP |
||||||
|
"flutter_engine.cc" |
||||||
|
"flutter_view_controller.cc" |
||||||
|
) |
||||||
|
list(TRANSFORM CPP_WRAPPER_SOURCES_APP PREPEND "${WRAPPER_ROOT}/") |
||||||
|
|
||||||
|
# Wrapper sources needed for a plugin. |
||||||
|
add_library(flutter_wrapper_plugin STATIC |
||||||
|
${CPP_WRAPPER_SOURCES_CORE} |
||||||
|
${CPP_WRAPPER_SOURCES_PLUGIN} |
||||||
|
) |
||||||
|
apply_standard_settings(flutter_wrapper_plugin) |
||||||
|
set_target_properties(flutter_wrapper_plugin PROPERTIES |
||||||
|
POSITION_INDEPENDENT_CODE ON) |
||||||
|
set_target_properties(flutter_wrapper_plugin PROPERTIES |
||||||
|
CXX_VISIBILITY_PRESET hidden) |
||||||
|
target_link_libraries(flutter_wrapper_plugin PUBLIC flutter) |
||||||
|
target_include_directories(flutter_wrapper_plugin PUBLIC |
||||||
|
"${WRAPPER_ROOT}/include" |
||||||
|
) |
||||||
|
add_dependencies(flutter_wrapper_plugin flutter_assemble) |
||||||
|
|
||||||
|
# Wrapper sources needed for the runner. |
||||||
|
add_library(flutter_wrapper_app STATIC |
||||||
|
${CPP_WRAPPER_SOURCES_CORE} |
||||||
|
${CPP_WRAPPER_SOURCES_APP} |
||||||
|
) |
||||||
|
apply_standard_settings(flutter_wrapper_app) |
||||||
|
target_link_libraries(flutter_wrapper_app PUBLIC flutter) |
||||||
|
target_include_directories(flutter_wrapper_app PUBLIC |
||||||
|
"${WRAPPER_ROOT}/include" |
||||||
|
) |
||||||
|
add_dependencies(flutter_wrapper_app flutter_assemble) |
||||||
|
|
||||||
|
# === Flutter tool backend === |
||||||
|
# _phony_ is a non-existent file to force this command to run every time, |
||||||
|
# since currently there's no way to get a full input/output list from the |
||||||
|
# flutter tool. |
||||||
|
set(PHONY_OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/_phony_") |
||||||
|
set_source_files_properties("${PHONY_OUTPUT}" PROPERTIES SYMBOLIC TRUE) |
||||||
|
add_custom_command( |
||||||
|
OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} |
||||||
|
${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} |
||||||
|
${CPP_WRAPPER_SOURCES_APP} |
||||||
|
${PHONY_OUTPUT} |
||||||
|
COMMAND ${CMAKE_COMMAND} -E env |
||||||
|
${FLUTTER_TOOL_ENVIRONMENT} |
||||||
|
"${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.bat" |
||||||
|
windows-x64 $<CONFIG> |
||||||
|
VERBATIM |
||||||
|
) |
||||||
|
add_custom_target(flutter_assemble DEPENDS |
||||||
|
"${FLUTTER_LIBRARY}" |
||||||
|
${FLUTTER_LIBRARY_HEADERS} |
||||||
|
${CPP_WRAPPER_SOURCES_CORE} |
||||||
|
${CPP_WRAPPER_SOURCES_PLUGIN} |
||||||
|
${CPP_WRAPPER_SOURCES_APP} |
||||||
|
) |
@ -0,0 +1,17 @@ |
|||||||
|
//
|
||||||
|
// Generated file. Do not edit.
|
||||||
|
//
|
||||||
|
|
||||||
|
// clang-format off
|
||||||
|
|
||||||
|
#include "generated_plugin_registrant.h" |
||||||
|
|
||||||
|
#include <dynamic_color/dynamic_color_plugin_c_api.h> |
||||||
|
#include <url_launcher_windows/url_launcher_windows.h> |
||||||
|
|
||||||
|
void RegisterPlugins(flutter::PluginRegistry* registry) { |
||||||
|
DynamicColorPluginCApiRegisterWithRegistrar( |
||||||
|
registry->GetRegistrarForPlugin("DynamicColorPluginCApi")); |
||||||
|
UrlLauncherWindowsRegisterWithRegistrar( |
||||||
|
registry->GetRegistrarForPlugin("UrlLauncherWindows")); |
||||||
|
} |
@ -0,0 +1,15 @@ |
|||||||
|
//
|
||||||
|
// Generated file. Do not edit.
|
||||||
|
//
|
||||||
|
|
||||||
|
// clang-format off
|
||||||
|
|
||||||
|
#ifndef GENERATED_PLUGIN_REGISTRANT_ |
||||||
|
#define GENERATED_PLUGIN_REGISTRANT_ |
||||||
|
|
||||||
|
#include <flutter/plugin_registry.h> |
||||||
|
|
||||||
|
// Registers Flutter plugins.
|
||||||
|
void RegisterPlugins(flutter::PluginRegistry* registry); |
||||||
|
|
||||||
|
#endif // GENERATED_PLUGIN_REGISTRANT_
|
@ -0,0 +1,25 @@ |
|||||||
|
# |
||||||
|
# Generated file, do not edit. |
||||||
|
# |
||||||
|
|
||||||
|
list(APPEND FLUTTER_PLUGIN_LIST |
||||||
|
dynamic_color |
||||||
|
url_launcher_windows |
||||||
|
) |
||||||
|
|
||||||
|
list(APPEND FLUTTER_FFI_PLUGIN_LIST |
||||||
|
) |
||||||
|
|
||||||
|
set(PLUGIN_BUNDLED_LIBRARIES) |
||||||
|
|
||||||
|
foreach(plugin ${FLUTTER_PLUGIN_LIST}) |
||||||
|
add_subdirectory(flutter/ephemeral/.plugin_symlinks/${plugin}/windows plugins/${plugin}) |
||||||
|
target_link_libraries(${BINARY_NAME} PRIVATE ${plugin}_plugin) |
||||||
|
list(APPEND PLUGIN_BUNDLED_LIBRARIES $<TARGET_FILE:${plugin}_plugin>) |
||||||
|
list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${plugin}_bundled_libraries}) |
||||||
|
endforeach(plugin) |
||||||
|
|
||||||
|
foreach(ffi_plugin ${FLUTTER_FFI_PLUGIN_LIST}) |
||||||
|
add_subdirectory(flutter/ephemeral/.plugin_symlinks/${ffi_plugin}/windows plugins/${ffi_plugin}) |
||||||
|
list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${ffi_plugin}_bundled_libraries}) |
||||||
|
endforeach(ffi_plugin) |
@ -0,0 +1,39 @@ |
|||||||
|
cmake_minimum_required(VERSION 3.14) |
||||||
|
project(runner LANGUAGES CXX) |
||||||
|
|
||||||
|
# Define the application target. To change its name, change BINARY_NAME in the |
||||||
|
# top-level CMakeLists.txt, not the value here, or `flutter run` will no longer |
||||||
|
# work. |
||||||
|
# |
||||||
|
# Any new source files that you add to the application should be added here. |
||||||
|
add_executable(${BINARY_NAME} WIN32 |
||||||
|
"flutter_window.cpp" |
||||||
|
"main.cpp" |
||||||
|
"utils.cpp" |
||||||
|
"win32_window.cpp" |
||||||
|
"${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" |
||||||
|
"Runner.rc" |
||||||
|
"runner.exe.manifest" |
||||||
|
) |
||||||
|
|
||||||
|
# Apply the standard set of build settings. This can be removed for applications |
||||||
|
# that need different build settings. |
||||||
|
apply_standard_settings(${BINARY_NAME}) |
||||||
|
|
||||||
|
# Add preprocessor definitions for the build version. |
||||||
|
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION=\"${FLUTTER_VERSION}\"") |
||||||
|
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MAJOR=${FLUTTER_VERSION_MAJOR}") |
||||||
|
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MINOR=${FLUTTER_VERSION_MINOR}") |
||||||
|
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_PATCH=${FLUTTER_VERSION_PATCH}") |
||||||
|
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_BUILD=${FLUTTER_VERSION_BUILD}") |
||||||
|
|
||||||
|
# Disable Windows macros that collide with C++ standard library functions. |
||||||
|
target_compile_definitions(${BINARY_NAME} PRIVATE "NOMINMAX") |
||||||
|
|
||||||
|
# Add dependency libraries and include directories. Add any application-specific |
||||||
|
# dependencies here. |
||||||
|
target_link_libraries(${BINARY_NAME} PRIVATE flutter flutter_wrapper_app) |
||||||
|
target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") |
||||||
|
|
||||||
|
# Run the Flutter tool portions of the build. This must not be removed. |
||||||
|
add_dependencies(${BINARY_NAME} flutter_assemble) |
@ -0,0 +1,121 @@ |
|||||||
|
// Microsoft Visual C++ generated resource script. |
||||||
|
// |
||||||
|
#pragma code_page(65001) |
||||||
|
#include "resource.h" |
||||||
|
|
||||||
|
#define APSTUDIO_READONLY_SYMBOLS |
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
// |
||||||
|
// Generated from the TEXTINCLUDE 2 resource. |
||||||
|
// |
||||||
|
#include "winres.h" |
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
#undef APSTUDIO_READONLY_SYMBOLS |
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
// English (United States) resources |
||||||
|
|
||||||
|
#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU) |
||||||
|
LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US |
||||||
|
|
||||||
|
#ifdef APSTUDIO_INVOKED |
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
// |
||||||
|
// TEXTINCLUDE |
||||||
|
// |
||||||
|
|
||||||
|
1 TEXTINCLUDE |
||||||
|
BEGIN |
||||||
|
"resource.h\0" |
||||||
|
END |
||||||
|
|
||||||
|
2 TEXTINCLUDE |
||||||
|
BEGIN |
||||||
|
"#include ""winres.h""\r\n" |
||||||
|
"\0" |
||||||
|
END |
||||||
|
|
||||||
|
3 TEXTINCLUDE |
||||||
|
BEGIN |
||||||
|
"\r\n" |
||||||
|
"\0" |
||||||
|
END |
||||||
|
|
||||||
|
#endif // APSTUDIO_INVOKED |
||||||
|
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
// |
||||||
|
// Icon |
||||||
|
// |
||||||
|
|
||||||
|
// Icon with lowest ID value placed first to ensure application icon |
||||||
|
// remains consistent on all systems. |
||||||
|
IDI_APP_ICON ICON "resources\\app_icon.ico" |
||||||
|
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
// |
||||||
|
// Version |
||||||
|
// |
||||||
|
|
||||||
|
#if defined(FLUTTER_VERSION_MAJOR) && defined(FLUTTER_VERSION_MINOR) && defined(FLUTTER_VERSION_PATCH) && defined(FLUTTER_VERSION_BUILD) |
||||||
|
#define VERSION_AS_NUMBER FLUTTER_VERSION_MAJOR,FLUTTER_VERSION_MINOR,FLUTTER_VERSION_PATCH,FLUTTER_VERSION_BUILD |
||||||
|
#else |
||||||
|
#define VERSION_AS_NUMBER 1,0,0,0 |
||||||
|
#endif |
||||||
|
|
||||||
|
#if defined(FLUTTER_VERSION) |
||||||
|
#define VERSION_AS_STRING FLUTTER_VERSION |
||||||
|
#else |
||||||
|
#define VERSION_AS_STRING "1.0.0" |
||||||
|
#endif |
||||||
|
|
||||||
|
VS_VERSION_INFO VERSIONINFO |
||||||
|
FILEVERSION VERSION_AS_NUMBER |
||||||
|
PRODUCTVERSION VERSION_AS_NUMBER |
||||||
|
FILEFLAGSMASK VS_FFI_FILEFLAGSMASK |
||||||
|
#ifdef _DEBUG |
||||||
|
FILEFLAGS VS_FF_DEBUG |
||||||
|
#else |
||||||
|
FILEFLAGS 0x0L |
||||||
|
#endif |
||||||
|
FILEOS VOS__WINDOWS32 |
||||||
|
FILETYPE VFT_APP |
||||||
|
FILESUBTYPE 0x0L |
||||||
|
BEGIN |
||||||
|
BLOCK "StringFileInfo" |
||||||
|
BEGIN |
||||||
|
BLOCK "040904e4" |
||||||
|
BEGIN |
||||||
|
VALUE "CompanyName", "ro.dcdev" "\0" |
||||||
|
VALUE "FileDescription", "info_tren" "\0" |
||||||
|
VALUE "FileVersion", VERSION_AS_STRING "\0" |
||||||
|
VALUE "InternalName", "info_tren" "\0" |
||||||
|
VALUE "LegalCopyright", "Copyright (C) 2022 ro.dcdev. All rights reserved." "\0" |
||||||
|
VALUE "OriginalFilename", "info_tren.exe" "\0" |
||||||
|
VALUE "ProductName", "info_tren" "\0" |
||||||
|
VALUE "ProductVersion", VERSION_AS_STRING "\0" |
||||||
|
END |
||||||
|
END |
||||||
|
BLOCK "VarFileInfo" |
||||||
|
BEGIN |
||||||
|
VALUE "Translation", 0x409, 1252 |
||||||
|
END |
||||||
|
END |
||||||
|
|
||||||
|
#endif // English (United States) resources |
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef APSTUDIO_INVOKED |
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
// |
||||||
|
// Generated from the TEXTINCLUDE 3 resource. |
||||||
|
// |
||||||
|
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////// |
||||||
|
#endif // not APSTUDIO_INVOKED |
@ -0,0 +1,61 @@ |
|||||||
|
#include "flutter_window.h" |
||||||
|
|
||||||
|
#include <optional> |
||||||
|
|
||||||
|
#include "flutter/generated_plugin_registrant.h" |
||||||
|
|
||||||
|
FlutterWindow::FlutterWindow(const flutter::DartProject& project) |
||||||
|
: project_(project) {} |
||||||
|
|
||||||
|
FlutterWindow::~FlutterWindow() {} |
||||||
|
|
||||||
|
bool FlutterWindow::OnCreate() { |
||||||
|
if (!Win32Window::OnCreate()) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
RECT frame = GetClientArea(); |
||||||
|
|
||||||
|
// The size here must match the window dimensions to avoid unnecessary surface
|
||||||
|
// creation / destruction in the startup path.
|
||||||
|
flutter_controller_ = std::make_unique<flutter::FlutterViewController>( |
||||||
|
frame.right - frame.left, frame.bottom - frame.top, project_); |
||||||
|
// Ensure that basic setup of the controller was successful.
|
||||||
|
if (!flutter_controller_->engine() || !flutter_controller_->view()) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
RegisterPlugins(flutter_controller_->engine()); |
||||||
|
SetChildContent(flutter_controller_->view()->GetNativeWindow()); |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
void FlutterWindow::OnDestroy() { |
||||||
|
if (flutter_controller_) { |
||||||
|
flutter_controller_ = nullptr; |
||||||
|
} |
||||||
|
|
||||||
|
Win32Window::OnDestroy(); |
||||||
|
} |
||||||
|
|
||||||
|
LRESULT |
||||||
|
FlutterWindow::MessageHandler(HWND hwnd, UINT const message, |
||||||
|
WPARAM const wparam, |
||||||
|
LPARAM const lparam) noexcept { |
||||||
|
// Give Flutter, including plugins, an opportunity to handle window messages.
|
||||||
|
if (flutter_controller_) { |
||||||
|
std::optional<LRESULT> result = |
||||||
|
flutter_controller_->HandleTopLevelWindowProc(hwnd, message, wparam, |
||||||
|
lparam); |
||||||
|
if (result) { |
||||||
|
return *result; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
switch (message) { |
||||||
|
case WM_FONTCHANGE: |
||||||
|
flutter_controller_->engine()->ReloadSystemFonts(); |
||||||
|
break; |
||||||
|
} |
||||||
|
|
||||||
|
return Win32Window::MessageHandler(hwnd, message, wparam, lparam); |
||||||
|
} |
@ -0,0 +1,33 @@ |
|||||||
|
#ifndef RUNNER_FLUTTER_WINDOW_H_ |
||||||
|
#define RUNNER_FLUTTER_WINDOW_H_ |
||||||
|
|
||||||
|
#include <flutter/dart_project.h> |
||||||
|
#include <flutter/flutter_view_controller.h> |
||||||
|
|
||||||
|
#include <memory> |
||||||
|
|
||||||
|
#include "win32_window.h" |
||||||
|
|
||||||
|
// A window that does nothing but host a Flutter view.
|
||||||
|
class FlutterWindow : public Win32Window { |
||||||
|
public: |
||||||
|
// Creates a new FlutterWindow hosting a Flutter view running |project|.
|
||||||
|
explicit FlutterWindow(const flutter::DartProject& project); |
||||||
|
virtual ~FlutterWindow(); |
||||||
|
|
||||||
|
protected: |
||||||
|
// Win32Window:
|
||||||
|
bool OnCreate() override; |
||||||
|
void OnDestroy() override; |
||||||
|
LRESULT MessageHandler(HWND window, UINT const message, WPARAM const wparam, |
||||||
|
LPARAM const lparam) noexcept override; |
||||||
|
|
||||||
|
private: |
||||||
|
// The project to run.
|
||||||
|
flutter::DartProject project_; |
||||||
|
|
||||||
|
// The Flutter instance hosted by this window.
|
||||||
|
std::unique_ptr<flutter::FlutterViewController> flutter_controller_; |
||||||
|
}; |
||||||
|
|
||||||
|
#endif // RUNNER_FLUTTER_WINDOW_H_
|
@ -0,0 +1,43 @@ |
|||||||
|
#include <flutter/dart_project.h> |
||||||
|
#include <flutter/flutter_view_controller.h> |
||||||
|
#include <windows.h> |
||||||
|
|
||||||
|
#include "flutter_window.h" |
||||||
|
#include "utils.h" |
||||||
|
|
||||||
|
int APIENTRY wWinMain(_In_ HINSTANCE instance, _In_opt_ HINSTANCE prev, |
||||||
|
_In_ wchar_t *command_line, _In_ int show_command) { |
||||||
|
// Attach to console when present (e.g., 'flutter run') or create a
|
||||||
|
// new console when running with a debugger.
|
||||||
|
if (!::AttachConsole(ATTACH_PARENT_PROCESS) && ::IsDebuggerPresent()) { |
||||||
|
CreateAndAttachConsole(); |
||||||
|
} |
||||||
|
|
||||||
|
// Initialize COM, so that it is available for use in the library and/or
|
||||||
|
// plugins.
|
||||||
|
::CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED); |
||||||
|
|
||||||
|
flutter::DartProject project(L"data"); |
||||||
|
|
||||||
|
std::vector<std::string> command_line_arguments = |
||||||
|
GetCommandLineArguments(); |
||||||
|
|
||||||
|
project.set_dart_entrypoint_arguments(std::move(command_line_arguments)); |
||||||
|
|
||||||
|
FlutterWindow window(project); |
||||||
|
Win32Window::Point origin(10, 10); |
||||||
|
Win32Window::Size size(1280, 720); |
||||||
|
if (!window.CreateAndShow(L"Info Tren", origin, size)) { |
||||||
|
return EXIT_FAILURE; |
||||||
|
} |
||||||
|
window.SetQuitOnClose(true); |
||||||
|
|
||||||
|
::MSG msg; |
||||||
|
while (::GetMessage(&msg, nullptr, 0, 0)) { |
||||||
|
::TranslateMessage(&msg); |
||||||
|
::DispatchMessage(&msg); |
||||||
|
} |
||||||
|
|
||||||
|
::CoUninitialize(); |
||||||
|
return EXIT_SUCCESS; |
||||||
|
} |
@ -0,0 +1,16 @@ |
|||||||
|
//{{NO_DEPENDENCIES}}
|
||||||
|
// Microsoft Visual C++ generated include file.
|
||||||
|
// Used by Runner.rc
|
||||||
|
//
|
||||||
|
#define IDI_APP_ICON 101 |
||||||
|
|
||||||
|
// Next default values for new objects
|
||||||
|
//
|
||||||
|
#ifdef APSTUDIO_INVOKED |
||||||
|
#ifndef APSTUDIO_READONLY_SYMBOLS |
||||||
|
#define _APS_NEXT_RESOURCE_VALUE 102 |
||||||
|
#define _APS_NEXT_COMMAND_VALUE 40001 |
||||||
|
#define _APS_NEXT_CONTROL_VALUE 1001 |
||||||
|
#define _APS_NEXT_SYMED_VALUE 101 |
||||||
|
#endif |
||||||
|
#endif |
After Width: | Height: | Size: 33 KiB |
@ -0,0 +1,20 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="yes"?> |
||||||
|
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0"> |
||||||
|
<application xmlns="urn:schemas-microsoft-com:asm.v3"> |
||||||
|
<windowsSettings> |
||||||
|
<dpiAwareness xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">PerMonitorV2</dpiAwareness> |
||||||
|
</windowsSettings> |
||||||
|
</application> |
||||||
|
<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1"> |
||||||
|
<application> |
||||||
|
<!-- Windows 10 and Windows 11 --> |
||||||
|
<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}"/> |
||||||
|
<!-- Windows 8.1 --> |
||||||
|
<supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}"/> |
||||||
|
<!-- Windows 8 --> |
||||||
|
<supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}"/> |
||||||
|
<!-- Windows 7 --> |
||||||
|
<supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}"/> |
||||||
|
</application> |
||||||
|
</compatibility> |
||||||
|
</assembly> |
@ -0,0 +1,64 @@ |
|||||||
|
#include "utils.h" |
||||||
|
|
||||||
|
#include <flutter_windows.h> |
||||||
|
#include <io.h> |
||||||
|
#include <stdio.h> |
||||||
|
#include <windows.h> |
||||||
|
|
||||||
|
#include <iostream> |
||||||
|
|
||||||
|
void CreateAndAttachConsole() { |
||||||
|
if (::AllocConsole()) { |
||||||
|
FILE *unused; |
||||||
|
if (freopen_s(&unused, "CONOUT$", "w", stdout)) { |
||||||
|
_dup2(_fileno(stdout), 1); |
||||||
|
} |
||||||
|
if (freopen_s(&unused, "CONOUT$", "w", stderr)) { |
||||||
|
_dup2(_fileno(stdout), 2); |
||||||
|
} |
||||||
|
std::ios::sync_with_stdio(); |
||||||
|
FlutterDesktopResyncOutputStreams(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
std::vector<std::string> GetCommandLineArguments() { |
||||||
|
// Convert the UTF-16 command line arguments to UTF-8 for the Engine to use.
|
||||||
|
int argc; |
||||||
|
wchar_t** argv = ::CommandLineToArgvW(::GetCommandLineW(), &argc); |
||||||
|
if (argv == nullptr) { |
||||||
|
return std::vector<std::string>(); |
||||||
|
} |
||||||
|
|
||||||
|
std::vector<std::string> command_line_arguments; |
||||||
|
|
||||||
|
// Skip the first argument as it's the binary name.
|
||||||
|
for (int i = 1; i < argc; i++) { |
||||||
|
command_line_arguments.push_back(Utf8FromUtf16(argv[i])); |
||||||
|
} |
||||||
|
|
||||||
|
::LocalFree(argv); |
||||||
|
|
||||||
|
return command_line_arguments; |
||||||
|
} |
||||||
|
|
||||||
|
std::string Utf8FromUtf16(const wchar_t* utf16_string) { |
||||||
|
if (utf16_string == nullptr) { |
||||||
|
return std::string(); |
||||||
|
} |
||||||
|
int target_length = ::WideCharToMultiByte( |
||||||
|
CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, |
||||||
|
-1, nullptr, 0, nullptr, nullptr); |
||||||
|
std::string utf8_string; |
||||||
|
if (target_length == 0 || target_length > utf8_string.max_size()) { |
||||||
|
return utf8_string; |
||||||
|
} |
||||||
|
utf8_string.resize(target_length); |
||||||
|
int converted_length = ::WideCharToMultiByte( |
||||||
|
CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, |
||||||
|
-1, utf8_string.data(), |
||||||
|
target_length, nullptr, nullptr); |
||||||
|
if (converted_length == 0) { |
||||||
|
return std::string(); |
||||||
|
} |
||||||
|
return utf8_string; |
||||||
|
} |
@ -0,0 +1,19 @@ |
|||||||
|
#ifndef RUNNER_UTILS_H_ |
||||||
|
#define RUNNER_UTILS_H_ |
||||||
|
|
||||||
|
#include <string> |
||||||
|
#include <vector> |
||||||
|
|
||||||
|
// Creates a console for the process, and redirects stdout and stderr to
|
||||||
|
// it for both the runner and the Flutter library.
|
||||||
|
void CreateAndAttachConsole(); |
||||||
|
|
||||||
|
// Takes a null-terminated wchar_t* encoded in UTF-16 and returns a std::string
|
||||||
|
// encoded in UTF-8. Returns an empty std::string on failure.
|
||||||
|
std::string Utf8FromUtf16(const wchar_t* utf16_string); |
||||||
|
|
||||||
|
// Gets the command line arguments passed in as a std::vector<std::string>,
|
||||||
|
// encoded in UTF-8. Returns an empty std::vector<std::string> on failure.
|
||||||
|
std::vector<std::string> GetCommandLineArguments(); |
||||||
|
|
||||||
|
#endif // RUNNER_UTILS_H_
|
@ -0,0 +1,245 @@ |
|||||||
|
#include "win32_window.h" |
||||||
|
|
||||||
|
#include <flutter_windows.h> |
||||||
|
|
||||||
|
#include "resource.h" |
||||||
|
|
||||||
|
namespace { |
||||||
|
|
||||||
|
constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW"; |
||||||
|
|
||||||
|
// The number of Win32Window objects that currently exist.
|
||||||
|
static int g_active_window_count = 0; |
||||||
|
|
||||||
|
using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd); |
||||||
|
|
||||||
|
// Scale helper to convert logical scaler values to physical using passed in
|
||||||
|
// scale factor
|
||||||
|
int Scale(int source, double scale_factor) { |
||||||
|
return static_cast<int>(source * scale_factor); |
||||||
|
} |
||||||
|
|
||||||
|
// Dynamically loads the |EnableNonClientDpiScaling| from the User32 module.
|
||||||
|
// This API is only needed for PerMonitor V1 awareness mode.
|
||||||
|
void EnableFullDpiSupportIfAvailable(HWND hwnd) { |
||||||
|
HMODULE user32_module = LoadLibraryA("User32.dll"); |
||||||
|
if (!user32_module) { |
||||||
|
return; |
||||||
|
} |
||||||
|
auto enable_non_client_dpi_scaling = |
||||||
|
reinterpret_cast<EnableNonClientDpiScaling*>( |
||||||
|
GetProcAddress(user32_module, "EnableNonClientDpiScaling")); |
||||||
|
if (enable_non_client_dpi_scaling != nullptr) { |
||||||
|
enable_non_client_dpi_scaling(hwnd); |
||||||
|
FreeLibrary(user32_module); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
// Manages the Win32Window's window class registration.
|
||||||
|
class WindowClassRegistrar { |
||||||
|
public: |
||||||
|
~WindowClassRegistrar() = default; |
||||||
|
|
||||||
|
// Returns the singleton registar instance.
|
||||||
|
static WindowClassRegistrar* GetInstance() { |
||||||
|
if (!instance_) { |
||||||
|
instance_ = new WindowClassRegistrar(); |
||||||
|
} |
||||||
|
return instance_; |
||||||
|
} |
||||||
|
|
||||||
|
// Returns the name of the window class, registering the class if it hasn't
|
||||||
|
// previously been registered.
|
||||||
|
const wchar_t* GetWindowClass(); |
||||||
|
|
||||||
|
// Unregisters the window class. Should only be called if there are no
|
||||||
|
// instances of the window.
|
||||||
|
void UnregisterWindowClass(); |
||||||
|
|
||||||
|
private: |
||||||
|
WindowClassRegistrar() = default; |
||||||
|
|
||||||
|
static WindowClassRegistrar* instance_; |
||||||
|
|
||||||
|
bool class_registered_ = false; |
||||||
|
}; |
||||||
|
|
||||||
|
WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr; |
||||||
|
|
||||||
|
const wchar_t* WindowClassRegistrar::GetWindowClass() { |
||||||
|
if (!class_registered_) { |
||||||
|
WNDCLASS window_class{}; |
||||||
|
window_class.hCursor = LoadCursor(nullptr, IDC_ARROW); |
||||||
|
window_class.lpszClassName = kWindowClassName; |
||||||
|
window_class.style = CS_HREDRAW | CS_VREDRAW; |
||||||
|
window_class.cbClsExtra = 0; |
||||||
|
window_class.cbWndExtra = 0; |
||||||
|
window_class.hInstance = GetModuleHandle(nullptr); |
||||||
|
window_class.hIcon = |
||||||
|
LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON)); |
||||||
|
window_class.hbrBackground = 0; |
||||||
|
window_class.lpszMenuName = nullptr; |
||||||
|
window_class.lpfnWndProc = Win32Window::WndProc; |
||||||
|
RegisterClass(&window_class); |
||||||
|
class_registered_ = true; |
||||||
|
} |
||||||
|
return kWindowClassName; |
||||||
|
} |
||||||
|
|
||||||
|
void WindowClassRegistrar::UnregisterWindowClass() { |
||||||
|
UnregisterClass(kWindowClassName, nullptr); |
||||||
|
class_registered_ = false; |
||||||
|
} |
||||||
|
|
||||||
|
Win32Window::Win32Window() { |
||||||
|
++g_active_window_count; |
||||||
|
} |
||||||
|
|
||||||
|
Win32Window::~Win32Window() { |
||||||
|
--g_active_window_count; |
||||||
|
Destroy(); |
||||||
|
} |
||||||
|
|
||||||
|
bool Win32Window::CreateAndShow(const std::wstring& title, |
||||||
|
const Point& origin, |
||||||
|
const Size& size) { |
||||||
|
Destroy(); |
||||||
|
|
||||||
|
const wchar_t* window_class = |
||||||
|
WindowClassRegistrar::GetInstance()->GetWindowClass(); |
||||||
|
|
||||||
|
const POINT target_point = {static_cast<LONG>(origin.x), |
||||||
|
static_cast<LONG>(origin.y)}; |
||||||
|
HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST); |
||||||
|
UINT dpi = FlutterDesktopGetDpiForMonitor(monitor); |
||||||
|
double scale_factor = dpi / 96.0; |
||||||
|
|
||||||
|
HWND window = CreateWindow( |
||||||
|
window_class, title.c_str(), WS_OVERLAPPEDWINDOW | WS_VISIBLE, |
||||||
|
Scale(origin.x, scale_factor), Scale(origin.y, scale_factor), |
||||||
|
Scale(size.width, scale_factor), Scale(size.height, scale_factor), |
||||||
|
nullptr, nullptr, GetModuleHandle(nullptr), this); |
||||||
|
|
||||||
|
if (!window) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
return OnCreate(); |
||||||
|
} |
||||||
|
|
||||||
|
// static
|
||||||
|
LRESULT CALLBACK Win32Window::WndProc(HWND const window, |
||||||
|
UINT const message, |
||||||
|
WPARAM const wparam, |
||||||
|
LPARAM const lparam) noexcept { |
||||||
|
if (message == WM_NCCREATE) { |
||||||
|
auto window_struct = reinterpret_cast<CREATESTRUCT*>(lparam); |
||||||
|
SetWindowLongPtr(window, GWLP_USERDATA, |
||||||
|
reinterpret_cast<LONG_PTR>(window_struct->lpCreateParams)); |
||||||
|
|
||||||
|
auto that = static_cast<Win32Window*>(window_struct->lpCreateParams); |
||||||
|
EnableFullDpiSupportIfAvailable(window); |
||||||
|
that->window_handle_ = window; |
||||||
|
} else if (Win32Window* that = GetThisFromHandle(window)) { |
||||||
|
return that->MessageHandler(window, message, wparam, lparam); |
||||||
|
} |
||||||
|
|
||||||
|
return DefWindowProc(window, message, wparam, lparam); |
||||||
|
} |
||||||
|
|
||||||
|
LRESULT |
||||||
|
Win32Window::MessageHandler(HWND hwnd, |
||||||
|
UINT const message, |
||||||
|
WPARAM const wparam, |
||||||
|
LPARAM const lparam) noexcept { |
||||||
|
switch (message) { |
||||||
|
case WM_DESTROY: |
||||||
|
window_handle_ = nullptr; |
||||||
|
Destroy(); |
||||||
|
if (quit_on_close_) { |
||||||
|
PostQuitMessage(0); |
||||||
|
} |
||||||
|
return 0; |
||||||
|
|
||||||
|
case WM_DPICHANGED: { |
||||||
|
auto newRectSize = reinterpret_cast<RECT*>(lparam); |
||||||
|
LONG newWidth = newRectSize->right - newRectSize->left; |
||||||
|
LONG newHeight = newRectSize->bottom - newRectSize->top; |
||||||
|
|
||||||
|
SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth, |
||||||
|
newHeight, SWP_NOZORDER | SWP_NOACTIVATE); |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
||||||
|
case WM_SIZE: { |
||||||
|
RECT rect = GetClientArea(); |
||||||
|
if (child_content_ != nullptr) { |
||||||
|
// Size and position the child window.
|
||||||
|
MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left, |
||||||
|
rect.bottom - rect.top, TRUE); |
||||||
|
} |
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
case WM_ACTIVATE: |
||||||
|
if (child_content_ != nullptr) { |
||||||
|
SetFocus(child_content_); |
||||||
|
} |
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
return DefWindowProc(window_handle_, message, wparam, lparam); |
||||||
|
} |
||||||
|
|
||||||
|
void Win32Window::Destroy() { |
||||||
|
OnDestroy(); |
||||||
|
|
||||||
|
if (window_handle_) { |
||||||
|
DestroyWindow(window_handle_); |
||||||
|
window_handle_ = nullptr; |
||||||
|
} |
||||||
|
if (g_active_window_count == 0) { |
||||||
|
WindowClassRegistrar::GetInstance()->UnregisterWindowClass(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept { |
||||||
|
return reinterpret_cast<Win32Window*>( |
||||||
|
GetWindowLongPtr(window, GWLP_USERDATA)); |
||||||
|
} |
||||||
|
|
||||||
|
void Win32Window::SetChildContent(HWND content) { |
||||||
|
child_content_ = content; |
||||||
|
SetParent(content, window_handle_); |
||||||
|
RECT frame = GetClientArea(); |
||||||
|
|
||||||
|
MoveWindow(content, frame.left, frame.top, frame.right - frame.left, |
||||||
|
frame.bottom - frame.top, true); |
||||||
|
|
||||||
|
SetFocus(child_content_); |
||||||
|
} |
||||||
|
|
||||||
|
RECT Win32Window::GetClientArea() { |
||||||
|
RECT frame; |
||||||
|
GetClientRect(window_handle_, &frame); |
||||||
|
return frame; |
||||||
|
} |
||||||
|
|
||||||
|
HWND Win32Window::GetHandle() { |
||||||
|
return window_handle_; |
||||||
|
} |
||||||
|
|
||||||
|
void Win32Window::SetQuitOnClose(bool quit_on_close) { |
||||||
|
quit_on_close_ = quit_on_close; |
||||||
|
} |
||||||
|
|
||||||
|
bool Win32Window::OnCreate() { |
||||||
|
// No-op; provided for subclasses.
|
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
void Win32Window::OnDestroy() { |
||||||
|
// No-op; provided for subclasses.
|
||||||
|
} |
@ -0,0 +1,98 @@ |
|||||||
|
#ifndef RUNNER_WIN32_WINDOW_H_ |
||||||
|
#define RUNNER_WIN32_WINDOW_H_ |
||||||
|
|
||||||
|
#include <windows.h> |
||||||
|
|
||||||
|
#include <functional> |
||||||
|
#include <memory> |
||||||
|
#include <string> |
||||||
|
|
||||||
|
// A class abstraction for a high DPI-aware Win32 Window. Intended to be
|
||||||
|
// inherited from by classes that wish to specialize with custom
|
||||||
|
// rendering and input handling
|
||||||
|
class Win32Window { |
||||||
|
public: |
||||||
|
struct Point { |
||||||
|
unsigned int x; |
||||||
|
unsigned int y; |
||||||
|
Point(unsigned int x, unsigned int y) : x(x), y(y) {} |
||||||
|
}; |
||||||
|
|
||||||
|
struct Size { |
||||||
|
unsigned int width; |
||||||
|
unsigned int height; |
||||||
|
Size(unsigned int width, unsigned int height) |
||||||
|
: width(width), height(height) {} |
||||||
|
}; |
||||||
|
|
||||||
|
Win32Window(); |
||||||
|
virtual ~Win32Window(); |
||||||
|
|
||||||
|
// Creates and shows a win32 window with |title| and position and size using
|
||||||
|
// |origin| and |size|. New windows are created on the default monitor. Window
|
||||||
|
// sizes are specified to the OS in physical pixels, hence to ensure a
|
||||||
|
// consistent size to will treat the width height passed in to this function
|
||||||
|
// as logical pixels and scale to appropriate for the default monitor. Returns
|
||||||
|
// true if the window was created successfully.
|
||||||
|
bool CreateAndShow(const std::wstring& title, |
||||||
|
const Point& origin, |
||||||
|
const Size& size); |
||||||
|
|
||||||
|
// Release OS resources associated with window.
|
||||||
|
void Destroy(); |
||||||
|
|
||||||
|
// Inserts |content| into the window tree.
|
||||||
|
void SetChildContent(HWND content); |
||||||
|
|
||||||
|
// Returns the backing Window handle to enable clients to set icon and other
|
||||||
|
// window properties. Returns nullptr if the window has been destroyed.
|
||||||
|
HWND GetHandle(); |
||||||
|
|
||||||
|
// If true, closing this window will quit the application.
|
||||||
|
void SetQuitOnClose(bool quit_on_close); |
||||||
|
|
||||||
|
// Return a RECT representing the bounds of the current client area.
|
||||||
|
RECT GetClientArea(); |
||||||
|
|
||||||
|
protected: |
||||||
|
// Processes and route salient window messages for mouse handling,
|
||||||
|
// size change and DPI. Delegates handling of these to member overloads that
|
||||||
|
// inheriting classes can handle.
|
||||||
|
virtual LRESULT MessageHandler(HWND window, |
||||||
|
UINT const message, |
||||||
|
WPARAM const wparam, |
||||||
|
LPARAM const lparam) noexcept; |
||||||
|
|
||||||
|
// Called when CreateAndShow is called, allowing subclass window-related
|
||||||
|
// setup. Subclasses should return false if setup fails.
|
||||||
|
virtual bool OnCreate(); |
||||||
|
|
||||||
|
// Called when Destroy is called.
|
||||||
|
virtual void OnDestroy(); |
||||||
|
|
||||||
|
private: |
||||||
|
friend class WindowClassRegistrar; |
||||||
|
|
||||||
|
// OS callback called by message pump. Handles the WM_NCCREATE message which
|
||||||
|
// is passed when the non-client area is being created and enables automatic
|
||||||
|
// non-client DPI scaling so that the non-client area automatically
|
||||||
|
// responsponds to changes in DPI. All other messages are handled by
|
||||||
|
// MessageHandler.
|
||||||
|
static LRESULT CALLBACK WndProc(HWND const window, |
||||||
|
UINT const message, |
||||||
|
WPARAM const wparam, |
||||||
|
LPARAM const lparam) noexcept; |
||||||
|
|
||||||
|
// Retrieves a class instance pointer for |window|
|
||||||
|
static Win32Window* GetThisFromHandle(HWND const window) noexcept; |
||||||
|
|
||||||
|
bool quit_on_close_ = false; |
||||||
|
|
||||||
|
// window handle for top level window.
|
||||||
|
HWND window_handle_ = nullptr; |
||||||
|
|
||||||
|
// window handle for hosted content.
|
||||||
|
HWND child_content_ = nullptr; |
||||||
|
}; |
||||||
|
|
||||||
|
#endif // RUNNER_WIN32_WINDOW_H_
|
Loading…
Reference in new issue