Skip to content

Commit 78bd670

Browse files
Handle contacts limited permission (#1446)
* Handle contacts limited permission (#1395) * Handle contacts limited permission * Version bump and comments * Resolve build errors and update example project * Update build configuration to use macos-latest * Only handle CNAuthorizationLimited on iOS 18+ * Force build workflow to use macOS 15 * Upgrade Example app --------- Co-authored-by: Hari07 <[email protected]>
1 parent c1a6a9b commit 78bd670

File tree

10 files changed

+195
-164
lines changed

10 files changed

+195
-164
lines changed

.github/workflows/permission_handler.yaml

+14-21
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,15 @@ name: permission_handler
66
# events but only for the main branch
77
on:
88
push:
9-
branches: [ main ]
9+
branches: [main]
1010
paths:
11-
- 'permission_handler/**'
12-
- '.github/workflows/permission_handler.yaml'
11+
- "permission_handler/**"
12+
- ".github/workflows/permission_handler.yaml"
1313
pull_request:
14-
branches: [ main ]
14+
branches: [main]
1515
paths:
16-
- 'permission_handler/**'
17-
- '.github/workflows/permission_handler.yaml'
16+
- "permission_handler/**"
17+
- ".github/workflows/permission_handler.yaml"
1818

1919
# A workflow run is made up of one or more jobs that can run sequentially or in parallel
2020
jobs:
@@ -23,37 +23,30 @@ jobs:
2323

2424
# The type of runner that the job will run on
2525
#
26-
# TODO(mvanbeusekom): Manually set to macOS 13 to support Xcode 15 and iOS 17 SDKs.
27-
# Currently `macos-latest` is based on macOS 12 and doesn't support iOS 17 SDK. This
26+
# TODO(mvanbeusekom): Manually set to macOS 15 to support Xcode 16 and iOS 18 SDKs.
27+
# Currently `macos-latest` is based on macOS 14 and doesn't support iOS 18 SDK. This
2828
# should be moved back to `macos-latest` when GitHub Actions images are updated.
29-
runs-on: macos-13
29+
runs-on: macos-15
3030

3131
env:
32-
source-directory: ./permission_handler
32+
source-directory: ./permission_handler
3333
example-directory: ./permission_handler/example
3434

3535
# Steps represent a sequence of tasks that will be executed as part of the job
3636
steps:
3737
# Checks-out your repository under $GITHUB_WORKSPACE, so your job can access it
3838
- uses: actions/checkout@v3
3939

40-
# Override current Xcode version with version 15.0.1.
41-
#
42-
# TODO(mvanbeusekom): Remove when the macos-latest image supports version 15.0.1
43-
# out of the box (see https://github.com/actions/runner-images/blob/main/README.md).
44-
- name: Select Xcode version
45-
run: sudo xcode-select -s '/Applications/Xcode_15.0.1.app/Contents/Developer'
46-
4740
# Make sure JAVA version 17 is installed on build agent.
4841
- uses: actions/setup-java@v3
4942
with:
50-
distribution: 'temurin' # See 'Supported distributions' for available options
51-
java-version: '17'
43+
distribution: "temurin" # See 'Supported distributions' for available options
44+
java-version: "17"
5245

5346
# Make sure the stable version of Flutter is available
5447
- uses: subosito/flutter-action@v2
5548
with:
56-
channel: 'stable'
49+
channel: "stable"
5750

5851
# Download all Flutter packages
5952
- name: Download dependencies
@@ -64,7 +57,7 @@ jobs:
6457
- name: Run Dart Format
6558
run: dart format --set-exit-if-changed .
6659
working-directory: ${{env.source-directory}}
67-
60+
6861
# Run Flutter Analyzer
6962
- name: Run Flutter Analyzer
7063
run: flutter analyze

.github/workflows/permission_handler_apple.yaml

+11-19
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,15 @@ name: permission_handler_apple
66
# events but only for the main branch
77
on:
88
push:
9-
branches: [ main ]
9+
branches: [main]
1010
paths:
11-
- 'permission_handler_apple/**'
12-
- '.github/workflows/permission_handler_apple.yaml'
11+
- "permission_handler_apple/**"
12+
- ".github/workflows/permission_handler_apple.yaml"
1313
pull_request:
14-
branches: [ main ]
14+
branches: [main]
1515
paths:
16-
- 'permission_handler_apple/**'
17-
- '.github/workflows/permission_handler_apple.yaml'
16+
- "permission_handler_apple/**"
17+
- ".github/workflows/permission_handler_apple.yaml"
1818

1919
# A workflow run is made up of one or more jobs that can run sequentially or in parallel
2020
jobs:
@@ -23,31 +23,24 @@ jobs:
2323

2424
# The type of runner that the job will run on
2525
#
26-
# TODO(mvanbeusekom): Manually set to macOS 13 to support Xcode 15 and iOS 17 SDKs.
27-
# Currently `macos-latest` is based on macOS 12 and doesn't support iOS 17 SDK. This
26+
# TODO(mvanbeusekom): Manually set to macOS 15 to support Xcode 16 and iOS 18 SDKs.
27+
# Currently `macos-latest` is based on macOS 14 and doesn't support iOS 18 SDK. This
2828
# should be moved back to `macos-latest` when GitHub Actions images are updated.
29-
runs-on: macos-13
29+
runs-on: macos-15
3030

3131
env:
3232
source-directory: ./permission_handler_apple
33-
example-directory: ./permission_handler_apple/example
33+
example-directory: ./permission_handler_apple/example
3434

3535
# Steps represent a sequence of tasks that will be executed as part of the job
3636
steps:
3737
# Checks-out your repository under $GITHUB_WORKSPACE, so your job can access it
3838
- uses: actions/checkout@v3
3939

40-
# Override current Xcode version with version 15.0.1.
41-
#
42-
# TODO(mvanbeusekom): Remove when the macos-latest image supports version 15.0.1
43-
# out of the box (see https://github.com/actions/runner-images/blob/main/README.md).
44-
- name: Select Xcode version
45-
run: sudo xcode-select -s '/Applications/Xcode_15.0.1.app/Contents/Developer'
46-
4740
# Make sure the stable version of Flutter is available
4841
- uses: subosito/flutter-action@v2
4942
with:
50-
channel: 'stable'
43+
channel: "stable"
5144

5245
# Download all Flutter packages
5346
- name: Download dependencies
@@ -68,4 +61,3 @@ jobs:
6861
- name: Run iOS build
6962
run: flutter build ios --no-codesign --release
7063
working-directory: ${{env.example-directory}}
71-

permission_handler/example/lib/main.dart

+74-63
Original file line numberDiff line numberDiff line change
@@ -5,24 +5,32 @@ import 'package:flutter/material.dart';
55
import 'package:permission_handler/permission_handler.dart';
66

77
void main() {
8-
runApp(BaseflowPluginExample(
8+
runApp(
9+
BaseflowPluginExample(
910
pluginName: 'Permission Handler',
1011
githubURL: 'https://github.com/Baseflow/flutter-permission-handler',
1112
pubDevURL: 'https://pub.dev/packages/permission_handler',
12-
pages: [PermissionHandlerWidget.createPage()]));
13+
pages: [PermissionHandlerWidget.createPage()],
14+
),
15+
);
1316
}
1417

1518
///Defines the main theme color
1619
final MaterialColor themeMaterialColor =
1720
BaseflowPluginExample.createMaterialColor(
18-
const Color.fromRGBO(48, 49, 60, 1));
21+
const Color.fromRGBO(48, 49, 60, 1),
22+
);
1923

2024
/// A Flutter application demonstrating the functionality of this plugin
2125
class PermissionHandlerWidget extends StatefulWidget {
26+
const PermissionHandlerWidget._();
27+
2228
/// Create a page containing the functionality of this plugin
2329
static ExamplePage createPage() {
2430
return ExamplePage(
25-
Icons.location_on, (context) => PermissionHandlerWidget());
31+
Icons.location_on,
32+
(context) => const PermissionHandlerWidget._(),
33+
);
2634
}
2735

2836
@override
@@ -35,59 +43,61 @@ class _PermissionHandlerWidgetState extends State<PermissionHandlerWidget> {
3543
Widget build(BuildContext context) {
3644
return Center(
3745
child: ListView(
38-
children: Permission.values
39-
.where((permission) {
40-
if (Platform.isIOS) {
41-
return permission != Permission.unknown &&
42-
permission != Permission.phone &&
43-
permission != Permission.sms &&
44-
permission != Permission.ignoreBatteryOptimizations &&
45-
permission != Permission.accessMediaLocation &&
46-
permission != Permission.activityRecognition &&
47-
permission != Permission.manageExternalStorage &&
48-
permission != Permission.systemAlertWindow &&
49-
permission != Permission.requestInstallPackages &&
50-
permission != Permission.accessNotificationPolicy &&
51-
permission != Permission.bluetoothScan &&
52-
permission != Permission.bluetoothAdvertise &&
53-
permission != Permission.bluetoothConnect &&
54-
permission != Permission.nearbyWifiDevices &&
55-
permission != Permission.videos &&
56-
permission != Permission.audio &&
57-
permission != Permission.scheduleExactAlarm &&
58-
permission != Permission.sensorsAlways;
59-
} else {
60-
return permission != Permission.unknown &&
61-
permission != Permission.mediaLibrary &&
62-
permission != Permission.photosAddOnly &&
63-
permission != Permission.reminders &&
64-
permission != Permission.bluetooth &&
65-
permission != Permission.appTrackingTransparency &&
66-
permission != Permission.criticalAlerts &&
67-
permission != Permission.assistant;
68-
}
69-
})
70-
.map((permission) => PermissionWidget(permission))
71-
.toList()),
46+
children:
47+
Permission.values
48+
.where((permission) {
49+
if (Platform.isIOS) {
50+
return permission != Permission.unknown &&
51+
permission != Permission.phone &&
52+
permission != Permission.sms &&
53+
permission != Permission.ignoreBatteryOptimizations &&
54+
permission != Permission.accessMediaLocation &&
55+
permission != Permission.activityRecognition &&
56+
permission != Permission.manageExternalStorage &&
57+
permission != Permission.systemAlertWindow &&
58+
permission != Permission.requestInstallPackages &&
59+
permission != Permission.accessNotificationPolicy &&
60+
permission != Permission.bluetoothScan &&
61+
permission != Permission.bluetoothAdvertise &&
62+
permission != Permission.bluetoothConnect &&
63+
permission != Permission.nearbyWifiDevices &&
64+
permission != Permission.videos &&
65+
permission != Permission.audio &&
66+
permission != Permission.scheduleExactAlarm &&
67+
permission != Permission.sensorsAlways;
68+
} else {
69+
return permission != Permission.unknown &&
70+
permission != Permission.mediaLibrary &&
71+
permission != Permission.photosAddOnly &&
72+
permission != Permission.reminders &&
73+
permission != Permission.bluetooth &&
74+
permission != Permission.appTrackingTransparency &&
75+
permission != Permission.criticalAlerts &&
76+
permission != Permission.assistant;
77+
}
78+
})
79+
.map((permission) => PermissionWidget(permission))
80+
.toList(),
81+
),
7282
);
7383
}
7484
}
7585

7686
/// Permission widget containing information about the passed [Permission]
7787
class PermissionWidget extends StatefulWidget {
7888
/// Constructs a [PermissionWidget] for the supplied [Permission]
79-
const PermissionWidget(this._permission);
89+
const PermissionWidget(this.permission, {super.key});
8090

81-
final Permission _permission;
91+
/// The [Permission] for which this widget is rendered.
92+
final Permission permission;
8293

8394
@override
84-
_PermissionState createState() => _PermissionState(_permission);
95+
_PermissionState createState() => _PermissionState();
8596
}
8697

8798
class _PermissionState extends State<PermissionWidget> {
88-
_PermissionState(this._permission);
99+
_PermissionState();
89100

90-
final Permission _permission;
91101
PermissionStatus _permissionStatus = PermissionStatus.denied;
92102

93103
@override
@@ -98,7 +108,7 @@ class _PermissionState extends State<PermissionWidget> {
98108
}
99109

100110
void _listenForPermissionStatus() async {
101-
final status = await _permission.status;
111+
final status = await widget.permission.status;
102112
setState(() => _permissionStatus = status);
103113
}
104114

@@ -119,44 +129,45 @@ class _PermissionState extends State<PermissionWidget> {
119129
Widget build(BuildContext context) {
120130
return ListTile(
121131
title: Text(
122-
_permission.toString(),
132+
widget.permission.toString(),
123133
style: Theme.of(context).textTheme.bodyLarge,
124134
),
125135
subtitle: Text(
126136
_permissionStatus.toString(),
127137
style: TextStyle(color: getPermissionColor()),
128138
),
129-
trailing: (_permission is PermissionWithService)
130-
? IconButton(
131-
icon: const Icon(
132-
Icons.info,
133-
color: Colors.white,
134-
),
135-
onPressed: () {
136-
checkServiceStatus(
137-
context, _permission as PermissionWithService);
138-
})
139-
: null,
139+
trailing:
140+
(widget.permission is PermissionWithService)
141+
? IconButton(
142+
icon: const Icon(Icons.info, color: Colors.white),
143+
onPressed: () {
144+
checkServiceStatus(
145+
context,
146+
widget.permission as PermissionWithService,
147+
);
148+
},
149+
)
150+
: null,
140151
onTap: () {
141-
requestPermission(_permission);
152+
requestPermission(widget.permission);
142153
},
143154
);
144155
}
145156

146157
void checkServiceStatus(
147-
BuildContext context, PermissionWithService permission) async {
148-
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
149-
content: Text((await permission.serviceStatus).toString()),
150-
));
158+
BuildContext context,
159+
PermissionWithService permission,
160+
) async {
161+
ScaffoldMessenger.of(context).showSnackBar(
162+
SnackBar(content: Text((await permission.serviceStatus).toString())),
163+
);
151164
}
152165

153166
Future<void> requestPermission(Permission permission) async {
154167
final status = await permission.request();
155168

156169
setState(() {
157-
print(status);
158170
_permissionStatus = status;
159-
print(_permissionStatus);
160171
});
161172
}
162173
}

permission_handler/example/pubspec.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ name: permission_handler_example
22
description: Demonstrates how to use the permission_handler plugin.
33

44
environment:
5-
sdk: ">=2.15.0 <4.0.0"
5+
sdk: ^3.7.0
66

77
dependencies:
88
baseflow_plugin_template: ^2.1.1

permission_handler/lib/permission_handler.dart

+2-2
Original file line numberDiff line numberDiff line change
@@ -140,8 +140,8 @@ extension PermissionCheckShortcuts on Permission {
140140
/// *Only supported on iOS.*
141141
Future<bool> get isRestricted => status.isRestricted;
142142

143-
/// User has authorized this application for limited photo library access.
144-
/// *Only supported on iOS.(iOS14+)*
143+
/// User has authorized this application for limited access.
144+
/// *Only supported on iOS.(iOS14+ for photos, ios18+ for contacts)*
145145
Future<bool> get isLimited => status.isLimited;
146146

147147
/// Returns `true` when permissions are denied permanently.

permission_handler_apple/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 9.4.6
2+
3+
* Adds the ability to handle `CNAuthorizationStatusLimited` introduced in ios18
4+
15
## 9.4.5
26

37
* Fixes issue #1002, Xcode warning of the unresponsive of main thread when checking isLocationEnabled.

0 commit comments

Comments
 (0)