Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Handle contacts limited permission #1446

Merged
merged 6 commits into from
Feb 20, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
35 changes: 14 additions & 21 deletions .github/workflows/permission_handler.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,15 @@ name: permission_handler
# events but only for the main branch
on:
push:
branches: [ main ]
branches: [main]
paths:
- 'permission_handler/**'
- '.github/workflows/permission_handler.yaml'
- "permission_handler/**"
- ".github/workflows/permission_handler.yaml"
pull_request:
branches: [ main ]
branches: [main]
paths:
- 'permission_handler/**'
- '.github/workflows/permission_handler.yaml'
- "permission_handler/**"
- ".github/workflows/permission_handler.yaml"

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

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

env:
source-directory: ./permission_handler
source-directory: ./permission_handler
example-directory: ./permission_handler/example

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

# Override current Xcode version with version 15.0.1.
#
# TODO(mvanbeusekom): Remove when the macos-latest image supports version 15.0.1
# out of the box (see https://github.com/actions/runner-images/blob/main/README.md).
- name: Select Xcode version
run: sudo xcode-select -s '/Applications/Xcode_15.0.1.app/Contents/Developer'

# Make sure JAVA version 17 is installed on build agent.
- uses: actions/setup-java@v3
with:
distribution: 'temurin' # See 'Supported distributions' for available options
java-version: '17'
distribution: "temurin" # See 'Supported distributions' for available options
java-version: "17"

# Make sure the stable version of Flutter is available
- uses: subosito/flutter-action@v2
with:
channel: 'stable'
channel: "stable"

# Download all Flutter packages
- name: Download dependencies
Expand All @@ -64,7 +57,7 @@ jobs:
- name: Run Dart Format
run: dart format --set-exit-if-changed .
working-directory: ${{env.source-directory}}

# Run Flutter Analyzer
- name: Run Flutter Analyzer
run: flutter analyze
Expand Down
30 changes: 11 additions & 19 deletions .github/workflows/permission_handler_apple.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,15 @@ name: permission_handler_apple
# events but only for the main branch
on:
push:
branches: [ main ]
branches: [main]
paths:
- 'permission_handler_apple/**'
- '.github/workflows/permission_handler_apple.yaml'
- "permission_handler_apple/**"
- ".github/workflows/permission_handler_apple.yaml"
pull_request:
branches: [ main ]
branches: [main]
paths:
- 'permission_handler_apple/**'
- '.github/workflows/permission_handler_apple.yaml'
- "permission_handler_apple/**"
- ".github/workflows/permission_handler_apple.yaml"

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

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

env:
source-directory: ./permission_handler_apple
example-directory: ./permission_handler_apple/example
example-directory: ./permission_handler_apple/example

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

# Override current Xcode version with version 15.0.1.
#
# TODO(mvanbeusekom): Remove when the macos-latest image supports version 15.0.1
# out of the box (see https://github.com/actions/runner-images/blob/main/README.md).
- name: Select Xcode version
run: sudo xcode-select -s '/Applications/Xcode_15.0.1.app/Contents/Developer'

# Make sure the stable version of Flutter is available
- uses: subosito/flutter-action@v2
with:
channel: 'stable'
channel: "stable"

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

137 changes: 74 additions & 63 deletions permission_handler/example/lib/main.dart
Original file line number Diff line number Diff line change
Expand Up @@ -5,24 +5,32 @@ import 'package:flutter/material.dart';
import 'package:permission_handler/permission_handler.dart';

void main() {
runApp(BaseflowPluginExample(
runApp(
BaseflowPluginExample(
pluginName: 'Permission Handler',
githubURL: 'https://github.com/Baseflow/flutter-permission-handler',
pubDevURL: 'https://pub.dev/packages/permission_handler',
pages: [PermissionHandlerWidget.createPage()]));
pages: [PermissionHandlerWidget.createPage()],
),
);
}

///Defines the main theme color
final MaterialColor themeMaterialColor =
BaseflowPluginExample.createMaterialColor(
const Color.fromRGBO(48, 49, 60, 1));
const Color.fromRGBO(48, 49, 60, 1),
);

/// A Flutter application demonstrating the functionality of this plugin
class PermissionHandlerWidget extends StatefulWidget {
const PermissionHandlerWidget._();

/// Create a page containing the functionality of this plugin
static ExamplePage createPage() {
return ExamplePage(
Icons.location_on, (context) => PermissionHandlerWidget());
Icons.location_on,
(context) => const PermissionHandlerWidget._(),
);
}

@override
Expand All @@ -35,59 +43,61 @@ class _PermissionHandlerWidgetState extends State<PermissionHandlerWidget> {
Widget build(BuildContext context) {
return Center(
child: ListView(
children: Permission.values
.where((permission) {
if (Platform.isIOS) {
return permission != Permission.unknown &&
permission != Permission.phone &&
permission != Permission.sms &&
permission != Permission.ignoreBatteryOptimizations &&
permission != Permission.accessMediaLocation &&
permission != Permission.activityRecognition &&
permission != Permission.manageExternalStorage &&
permission != Permission.systemAlertWindow &&
permission != Permission.requestInstallPackages &&
permission != Permission.accessNotificationPolicy &&
permission != Permission.bluetoothScan &&
permission != Permission.bluetoothAdvertise &&
permission != Permission.bluetoothConnect &&
permission != Permission.nearbyWifiDevices &&
permission != Permission.videos &&
permission != Permission.audio &&
permission != Permission.scheduleExactAlarm &&
permission != Permission.sensorsAlways;
} else {
return permission != Permission.unknown &&
permission != Permission.mediaLibrary &&
permission != Permission.photosAddOnly &&
permission != Permission.reminders &&
permission != Permission.bluetooth &&
permission != Permission.appTrackingTransparency &&
permission != Permission.criticalAlerts &&
permission != Permission.assistant;
}
})
.map((permission) => PermissionWidget(permission))
.toList()),
children:
Permission.values
.where((permission) {
if (Platform.isIOS) {
return permission != Permission.unknown &&
permission != Permission.phone &&
permission != Permission.sms &&
permission != Permission.ignoreBatteryOptimizations &&
permission != Permission.accessMediaLocation &&
permission != Permission.activityRecognition &&
permission != Permission.manageExternalStorage &&
permission != Permission.systemAlertWindow &&
permission != Permission.requestInstallPackages &&
permission != Permission.accessNotificationPolicy &&
permission != Permission.bluetoothScan &&
permission != Permission.bluetoothAdvertise &&
permission != Permission.bluetoothConnect &&
permission != Permission.nearbyWifiDevices &&
permission != Permission.videos &&
permission != Permission.audio &&
permission != Permission.scheduleExactAlarm &&
permission != Permission.sensorsAlways;
} else {
return permission != Permission.unknown &&
permission != Permission.mediaLibrary &&
permission != Permission.photosAddOnly &&
permission != Permission.reminders &&
permission != Permission.bluetooth &&
permission != Permission.appTrackingTransparency &&
permission != Permission.criticalAlerts &&
permission != Permission.assistant;
}
})
.map((permission) => PermissionWidget(permission))
.toList(),
),
);
}
}

/// Permission widget containing information about the passed [Permission]
class PermissionWidget extends StatefulWidget {
/// Constructs a [PermissionWidget] for the supplied [Permission]
const PermissionWidget(this._permission);
const PermissionWidget(this.permission, {super.key});

final Permission _permission;
/// The [Permission] for which this widget is rendered.
final Permission permission;

@override
_PermissionState createState() => _PermissionState(_permission);
_PermissionState createState() => _PermissionState();
}

class _PermissionState extends State<PermissionWidget> {
_PermissionState(this._permission);
_PermissionState();

final Permission _permission;
PermissionStatus _permissionStatus = PermissionStatus.denied;

@override
Expand All @@ -98,7 +108,7 @@ class _PermissionState extends State<PermissionWidget> {
}

void _listenForPermissionStatus() async {
final status = await _permission.status;
final status = await widget.permission.status;
setState(() => _permissionStatus = status);
}

Expand All @@ -119,44 +129,45 @@ class _PermissionState extends State<PermissionWidget> {
Widget build(BuildContext context) {
return ListTile(
title: Text(
_permission.toString(),
widget.permission.toString(),
style: Theme.of(context).textTheme.bodyLarge,
),
subtitle: Text(
_permissionStatus.toString(),
style: TextStyle(color: getPermissionColor()),
),
trailing: (_permission is PermissionWithService)
? IconButton(
icon: const Icon(
Icons.info,
color: Colors.white,
),
onPressed: () {
checkServiceStatus(
context, _permission as PermissionWithService);
})
: null,
trailing:
(widget.permission is PermissionWithService)
? IconButton(
icon: const Icon(Icons.info, color: Colors.white),
onPressed: () {
checkServiceStatus(
context,
widget.permission as PermissionWithService,
);
},
)
: null,
onTap: () {
requestPermission(_permission);
requestPermission(widget.permission);
},
);
}

void checkServiceStatus(
BuildContext context, PermissionWithService permission) async {
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
content: Text((await permission.serviceStatus).toString()),
));
BuildContext context,
PermissionWithService permission,
) async {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text((await permission.serviceStatus).toString())),
);
}

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

setState(() {
print(status);
_permissionStatus = status;
print(_permissionStatus);
});
}
}
2 changes: 1 addition & 1 deletion permission_handler/example/pubspec.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ name: permission_handler_example
description: Demonstrates how to use the permission_handler plugin.

environment:
sdk: ">=2.15.0 <4.0.0"
sdk: ^3.7.0

dependencies:
baseflow_plugin_template: ^2.1.1
Expand Down
4 changes: 2 additions & 2 deletions permission_handler/lib/permission_handler.dart
Original file line number Diff line number Diff line change
Expand Up @@ -140,8 +140,8 @@ extension PermissionCheckShortcuts on Permission {
/// *Only supported on iOS.*
Future<bool> get isRestricted => status.isRestricted;

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

/// Returns `true` when permissions are denied permanently.
Expand Down
4 changes: 4 additions & 0 deletions permission_handler_apple/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
## 9.4.6

* Adds the ability to handle `CNAuthorizationStatusLimited` introduced in ios18

## 9.4.5

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