Skip to content

Commit 74cce8a

Browse files
committed
chore: update flutter, remove custom settings, build snap on github (#14)
* chore: update flutter and remove custom settings * chore: build snap on github
1 parent 8184194 commit 74cce8a

39 files changed

+786
-588
lines changed

.github/workflows/ci.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ on:
55
branches: [master]
66

77
env:
8-
FLUTTER_VERSION: '3.19.x'
8+
FLUTTER_VERSION: '3.27.4'
99

1010
jobs:
1111
analyze:
@@ -49,6 +49,6 @@ jobs:
4949
channel: 'stable'
5050
flutter-version: ${{env.FLUTTER_VERSION}}
5151
- run: sudo apt update
52-
- run: sudo apt install -y clang cmake curl libgtk-3-dev ninja-build pkg-config unzip libunwind-dev libgstreamer-plugins-base1.0-dev libgstreamer1.0-dev libmpv-dev
52+
- run: sudo apt install -y clang cmake curl libgtk-3-dev ninja-build pkg-config unzip libunwind-dev libsecret-1-dev
5353
- run: flutter pub get
5454
- run: flutter build linux -v

.github/workflows/snap.yaml

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
name: Create Snap Package
2+
3+
on:
4+
push:
5+
branches:
6+
- master
7+
workflow_dispatch:
8+
9+
env:
10+
FLUTTER_VERSION: "3.27.4"
11+
12+
jobs:
13+
build_and_release_linux_snap_edge_amd64:
14+
runs-on: ubuntu-latest
15+
steps:
16+
- uses: actions/checkout@v4
17+
with:
18+
fetch-depth: 5
19+
- uses: subosito/flutter-action@v2
20+
with:
21+
channel: "stable"
22+
flutter-version: ${{env.FLUTTER_VERSION}}
23+
- run: sudo apt update
24+
- run: sudo apt install -y clang cmake curl libgtk-3-dev ninja-build pkg-config unzip libunwind-dev libsecret-1-dev
25+
- run: flutter pub get
26+
- uses: snapcore/action-build@v1
27+
env:
28+
API_KEY: ${{ secrets.API_KEY }}
29+
id: build
30+
- uses: snapcore/action-publish@v1
31+
if: steps.build.outcome == 'success'
32+
env:
33+
SNAPCRAFT_STORE_CREDENTIALS: ${{ secrets.STORE_LOGIN }}
34+
with:
35+
snap: ${{ steps.build.outputs.snap }}
36+
release: edge
37+
38+
build_and_release_linux_snap_edge_arm64:
39+
runs-on: ubuntu-24.04-arm
40+
steps:
41+
- uses: actions/checkout@v4
42+
with:
43+
fetch-depth: 5
44+
- uses: subosito/flutter-action@v2
45+
with:
46+
channel: "main"
47+
flutter-version: ${{env.FLUTTER_VERSION}}
48+
- run: sudo apt update
49+
- run: sudo apt install -y clang cmake curl libgtk-3-dev ninja-build pkg-config unzip libunwind-dev libsecret-1-dev
50+
- run: flutter pub get
51+
- uses: snapcore/action-build@v1
52+
id: build
53+
env:
54+
API_KEY: ${{ secrets.API_KEY }}
55+
- uses: snapcore/action-publish@v1
56+
if: steps.build.outcome == 'success'
57+
env:
58+
SNAPCRAFT_STORE_CREDENTIALS: ${{ secrets.STORE_LOGIN }}
59+
with:
60+
snap: ${{ steps.build.outputs.snap }}
61+
release: edge

README.md

+15
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,13 @@
88

99
Requires an api key from [openweathermap](https://openweathermap.org) which you need to create yourself (free tier) in your own account.
1010

11+
1112
## Build
1213

14+
### Install libsecret
15+
16+
`sudo apt install libsecret-1-dev`
17+
1318
### Install Flutter
1419

1520
<details>
@@ -22,3 +27,13 @@ sudo apt -y install git curl cmake meson make clang libgtk-3-dev pkg-config && m
2227
```
2328

2429
</details>
30+
31+
### Run the app
32+
33+
For the first run or after every `flutter clean`, the app needs to be started with your API_KEY:
34+
35+
`flutter run --dart-define=API_KEY=YOUR_API_KEY_HERE_WITHOUT_QUOTES`
36+
37+
any other times you can use
38+
39+
`flutter run`

analysis_options.yaml

+6-1
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,9 @@ linter:
1010
require_trailing_commas: true
1111
use_super_parameters: true
1212
close_sinks: true
13-
prefer_relative_imports: true
13+
prefer_relative_imports: true
14+
sort_pub_dependencies: true
15+
unnecessary_parenthesis: true
16+
unnecessary_await_in_return: true
17+
unnecessary_late: true
18+
unnecessary_breaks: true

l10n.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
arb-dir: lib/src/l10n
1+
arb-dir: lib/l10n
22
template-arb-file: app_en.arb
33
output-localization-file: app_localizations.dart
44
nullable-getter: false

lib/app/app.dart

+133
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,133 @@
1+
import 'dart:io';
2+
import 'dart:ui';
3+
4+
import 'package:flutter/material.dart';
5+
import 'package:flutter_weather_bg_null_safety/bg/weather_bg.dart';
6+
import 'package:watch_it/watch_it.dart';
7+
import 'package:yaru/yaru.dart';
8+
9+
import '../constants.dart';
10+
import '../weather.dart';
11+
import '../extensions/build_context_x.dart';
12+
import '../l10n/l10n.dart';
13+
import '../weather/weather_model.dart';
14+
import 'side_bar.dart';
15+
16+
class App extends StatelessWidget {
17+
const App({super.key});
18+
19+
@override
20+
Widget build(BuildContext context) => MaterialApp(
21+
localizationsDelegates: AppLocalizations.localizationsDelegates,
22+
supportedLocales: supportedLocales,
23+
onGenerateTitle: (context) => 'MusicPod',
24+
debugShowCheckedModeBanner: false,
25+
theme: yaruLight,
26+
darkTheme: yaruDark.copyWith(
27+
tabBarTheme: TabBarTheme.of(context).copyWith(
28+
labelColor: Colors.white,
29+
unselectedLabelColor: Colors.white.withValues(
30+
alpha: 0.8,
31+
),
32+
),
33+
),
34+
home: const _StartPage(),
35+
scrollBehavior: const MaterialScrollBehavior().copyWith(
36+
dragDevices: {
37+
PointerDeviceKind.mouse,
38+
PointerDeviceKind.touch,
39+
PointerDeviceKind.stylus,
40+
PointerDeviceKind.unknown,
41+
PointerDeviceKind.trackpad,
42+
},
43+
),
44+
);
45+
}
46+
47+
class _StartPage extends StatefulWidget {
48+
const _StartPage();
49+
50+
@override
51+
State<_StartPage> createState() => _StartPageState();
52+
}
53+
54+
class _StartPageState extends State<_StartPage> {
55+
late final Future<void> _allReady;
56+
57+
@override
58+
void initState() {
59+
super.initState();
60+
_allReady = di.allReady();
61+
}
62+
63+
@override
64+
Widget build(BuildContext context) => FutureBuilder(
65+
future: _allReady,
66+
builder: (context, snapshot) =>
67+
snapshot.hasData ? const _AppPage() : const _LoadingPage(),
68+
);
69+
}
70+
71+
class _LoadingPage extends StatelessWidget {
72+
const _LoadingPage();
73+
74+
@override
75+
Widget build(BuildContext context) => const Scaffold(
76+
appBar: YaruWindowTitleBar(
77+
border: BorderSide.none,
78+
backgroundColor: Colors.transparent,
79+
),
80+
body: Center(
81+
child: YaruCircularProgressIndicator(),
82+
),
83+
);
84+
}
85+
86+
class _AppPage extends StatefulWidget with WatchItStatefulWidgetMixin {
87+
const _AppPage();
88+
89+
@override
90+
State<_AppPage> createState() => _AppPageState();
91+
}
92+
93+
class _AppPageState extends State<_AppPage> {
94+
@override
95+
void initState() {
96+
di<WeatherModel>().loadWeather();
97+
super.initState();
98+
}
99+
100+
@override
101+
Widget build(BuildContext context) {
102+
final weatherType = watchPropertyValue((WeatherModel m) => m.weatherType);
103+
104+
return LayoutBuilder(
105+
builder: (context, constraints) {
106+
var list = [
107+
if (constraints.maxWidth > kBreakPoint) const SideBar(),
108+
Expanded(
109+
child: WeatherPage(
110+
showDrawer: constraints.maxWidth < kBreakPoint,
111+
),
112+
),
113+
];
114+
return Stack(
115+
alignment: Alignment.center,
116+
children: [
117+
Opacity(
118+
opacity: Platform.isMacOS ? (context.light ? 1 : 0.6) : 0.7,
119+
child: WeatherBg(
120+
weatherType: weatherType,
121+
width: constraints.maxWidth,
122+
height: constraints.maxHeight,
123+
),
124+
),
125+
Row(
126+
children: Platform.isMacOS ? list.reversed.toList() : list,
127+
),
128+
],
129+
);
130+
},
131+
);
132+
}
133+
}
File renamed without changes.

lib/src/app/offline_page.dart lib/app/offline_page.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import 'package:flutter/material.dart';
22
import 'package:yaru/yaru.dart';
33

4-
import '../build_context_x.dart';
4+
import '../extensions/build_context_x.dart';
55
import '../l10n/l10n.dart';
66

77
class OfflinePage extends StatelessWidget {

lib/src/app/side_bar.dart lib/app/side_bar.dart

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,8 @@ import 'package:flutter/material.dart';
22
import 'package:watch_it/watch_it.dart';
33
import 'package:yaru/yaru.dart';
44

5-
import '../../constants.dart';
6-
import '../build_context_x.dart';
5+
import '../constants.dart';
6+
import '../extensions/build_context_x.dart';
77
import '../weather/view/city_search_field.dart';
88
import '../weather/weather_model.dart';
99

@@ -66,7 +66,7 @@ class SideBar extends StatelessWidget with WatchItMixin {
6666
);
6767

6868
return Material(
69-
color: theme.colorScheme.surface.withOpacity(0.4),
69+
color: theme.colorScheme.surface.withValues(alpha: 0.4),
7070
child: SizedBox(
7171
width: kPaneWidth,
7272
child: Column(
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.

lib/locations/locations_service.dart

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
import 'dart:async';
2+
3+
import '../settings/settings_service.dart';
4+
5+
class LocationsService {
6+
LocationsService({
7+
required SettingsService settingsService,
8+
}) : _settingsService = settingsService;
9+
final SettingsService _settingsService;
10+
11+
String? get lastLocation =>
12+
_settingsService.getString(key: SettingKeys.lastLocation);
13+
void setLastLocation(String? value) {
14+
if (value == null) return;
15+
_settingsService.setString(key: SettingKeys.lastLocation, value: value);
16+
}
17+
18+
Set<String> get favLocations =>
19+
_settingsService
20+
.getStrings(key: SettingKeys.favoriteLocations)
21+
?.toSet() ??
22+
{};
23+
bool isFavLocation(String value) => favLocations.contains(value);
24+
25+
void addFavLocation(String name) {
26+
if (favLocations.contains(name)) return;
27+
final favs = Set<String>.from(favLocations);
28+
favs.add(name);
29+
_settingsService.setStrings(
30+
key: SettingKeys.favoriteLocations,
31+
value: favs.toList(),
32+
);
33+
}
34+
35+
Future<void> removeFavLocation(String name) async {
36+
if (!favLocations.contains(name)) return;
37+
final favs = Set<String>.from(favLocations);
38+
favs.remove(name);
39+
_settingsService.setStrings(
40+
key: SettingKeys.favoriteLocations,
41+
value: favs.toList(),
42+
);
43+
}
44+
45+
Stream<bool> get propertiesChanged => _settingsService.propertiesChanged;
46+
}

lib/main.dart

+3-21
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,13 @@
11
import 'package:flutter/material.dart';
2-
import 'package:open_weather_client/open_weather.dart';
3-
import 'package:watch_it/watch_it.dart';
42
import 'package:yaru/yaru.dart';
53

6-
import 'src/app/app.dart';
7-
import 'src/app/app_model.dart';
8-
import 'src/locations/locations_service.dart';
9-
import 'src/weather/weather_model.dart';
4+
import 'app/app.dart';
5+
import 'register_dependencies.dart';
106

117
Future<void> main() async {
128
await YaruWindowTitleBar.ensureInitialized();
139

14-
final locationsService = LocationsService();
15-
await locationsService.init();
16-
di.registerSingleton<LocationsService>(
17-
locationsService,
18-
dispose: (s) => s.dispose(),
19-
);
20-
di.registerSingleton(AppModel());
21-
22-
di.registerLazySingleton(
23-
() => WeatherModel(
24-
locationsService: locationsService,
25-
openWeather: OpenWeather(apiKey: locationsService.apiKey ?? ''),
26-
),
27-
dispose: (s) => s.dispose(),
28-
);
10+
registerDependencies();
2911

3012
runApp(const App());
3113
}

0 commit comments

Comments
 (0)