Skip to content

Commit

Permalink
Fuse.Camera: update permissions for Android 13
Browse files Browse the repository at this point in the history
Co-authored-by: Morten Daniel Fornes <mdfornes@gmail.com>

This makes the Camera and CameraRoll modules work on Android 13.

Closes #1481
  • Loading branch information
ckarmy authored and mortend committed Sep 23, 2023
1 parent 51adf62 commit c00910c
Show file tree
Hide file tree
Showing 5 changed files with 151 additions and 39 deletions.
15 changes: 15 additions & 0 deletions Source/Fuse.Android.Permissions/AndroidPermissionsInternal.uno
Original file line number Diff line number Diff line change
Expand Up @@ -460,6 +460,21 @@ namespace Fuse.Android.Permissions.Internal
return new PlatformPermission("android.permission.READ_EXTERNAL_STORAGE");
}
[TargetSpecificImplementation]
internal static extern PlatformPermission _read_media_images()
{
return new PlatformPermission("android.permission.READ_MEDIA_IMAGES");
}
[TargetSpecificImplementation]
internal static extern PlatformPermission _read_media_audio()
{
return new PlatformPermission("android.permission.READ_MEDIA_AUDIO");
}
[TargetSpecificImplementation]
internal static extern PlatformPermission _read_media_video()
{
return new PlatformPermission("android.permission.READ_MEDIA_VIDEO");
}
[TargetSpecificImplementation]
internal static extern PlatformPermission _read_frame_buffer()
{
return new PlatformPermission("android.permission.READ_FRAME_BUFFER");
Expand Down
12 changes: 12 additions & 0 deletions Source/Fuse.Android.Permissions/AndroidPermissionsInternal.uxl
Original file line number Diff line number Diff line change
Expand Up @@ -364,6 +364,18 @@
<Require AndroidManifest.Permission="android.permission.READ_EXTERNAL_STORAGE" />
</Method>

<Method Signature="_read_media_images():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_MEDIA_IMAGES" />
</Method>

<Method Signature="_read_media_audio():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_MEDIA_AUDIO" />
</Method>

<Method Signature="_read_media_video():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_MEDIA_VIDEO" />
</Method>

<Method Signature="_read_frame_buffer():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_FRAME_BUFFER" />
</Method>
Expand Down
3 changes: 3 additions & 0 deletions Source/Fuse.Android.Permissions/Permissions.uno
Original file line number Diff line number Diff line change
Expand Up @@ -209,6 +209,9 @@ namespace Fuse.Android.Permissions
public static PlatformPermission READ_CALL_LOG { get { return Internal.Android._read_call_log(); } }
public static PlatformPermission READ_CONTACTS { get { return Internal.Android._read_contacts(); } }
public static PlatformPermission READ_EXTERNAL_STORAGE { get { return Internal.Android._read_external_storage(); } }
public static PlatformPermission READ_MEDIA_IMAGES { get { return Internal.Android._read_media_images(); } }
public static PlatformPermission READ_MEDIA_AUDIO { get { return Internal.Android._read_media_audio(); } }
public static PlatformPermission READ_MEDIA_VIDEO { get { return Internal.Android._read_media_video(); } }
public static PlatformPermission READ_FRAME_BUFFER { get { return Internal.Android._read_frame_buffer(); } }
public static PlatformPermission READ_HISTORY_BOOKMARKS { get { return Internal.Android._read_history_bookmarks(); } }
public static PlatformPermission READ_INPUT_STATE { get { return Internal.Android._read_input_state(); } }
Expand Down
12 changes: 12 additions & 0 deletions Source/Fuse.Android.Permissions/Permissions.uxl
Original file line number Diff line number Diff line change
Expand Up @@ -364,6 +364,18 @@
<Require AndroidManifest.Permission="android.permission.READ_EXTERNAL_STORAGE" />
</Method>

<Method Signature="_read_media_images():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_MEDIA_IMAGES" />
</Method>

<Method Signature="_read_media_audio():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_MEDIA_AUDIO" />
</Method>

<Method Signature="_read_media_video():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_MEDIA_VIDEO" />
</Method>

<Method Signature="_read_frame_buffer():Fuse.Android.Permissions.PlatformPermission">
<Require AndroidManifest.Permission="android.permission.READ_FRAME_BUFFER" />
</Method>
Expand Down
148 changes: 109 additions & 39 deletions Source/Fuse.Camera/Android/AndroidCamera.uno
Original file line number Diff line number Diff line change
Expand Up @@ -4,21 +4,36 @@ using Uno.Compiler.ExportTargetInterop;
using Android;
using Fuse.ImageTools;
using Fuse.Android.Permissions;

namespace Fuse.Camera
{

extern (Android) static internal class AndroidCamera
{
internal static void TakePicture(Promise<Image> p)
{
var permissions = new PlatformPermission[]
if (AndroidProperties.BuildVersion >= 33)
{
var permissions = new PlatformPermission[]
{
Permissions.Android.CAMERA,
Permissions.Android.READ_MEDIA_IMAGES,
Permissions.Android.READ_MEDIA_VIDEO,
Permissions.Android.READ_MEDIA_AUDIO
};

Permissions.Request(permissions).Then(new TakePictureCommand(p).Execute, p.Reject);
}
else
{
Permissions.Android.CAMERA,
Permissions.Android.WRITE_EXTERNAL_STORAGE,
Permissions.Android.READ_EXTERNAL_STORAGE
};
var permissions = new PlatformPermission[]
{
Permissions.Android.CAMERA,
Permissions.Android.WRITE_EXTERNAL_STORAGE,
Permissions.Android.READ_EXTERNAL_STORAGE
};

Permissions.Request(permissions).Then(new TakePictureCommand(p).Execute, p.Reject);
Permissions.Request(permissions).Then(new TakePictureCommand(p).Execute, p.Reject);
}
}

internal static void CheckPermissions(Promise<string> p)
Expand All @@ -32,15 +47,15 @@ namespace Fuse.Camera
}
}

[ForeignInclude(Language.Java,
"android.provider.MediaStore",
"com.fuse.Activity",
"com.fuse.camera.Image",
[ForeignInclude(Language.Java,
"android.provider.MediaStore",
"com.fuse.Activity",
"com.fuse.camera.Image",
"android.os.Build",
"androidx.core.content.FileProvider",
"androidx.core.content.FileProvider",
"java.io.File",
"android.net.Uri",
"android.util.Log",
"android.net.Uri",
"android.util.Log",
"android.content.Intent")]
extern (Android) internal class TakePictureCommand
{
Expand All @@ -51,18 +66,18 @@ namespace Fuse.Camera
}
public void Execute(PlatformPermission[] grantedPermissions)
{
if(grantedPermissions.Length<3)
if (grantedPermissions.Length < 3)
{
_promise.Reject(new Exception("Required permissions were not granted."));
return;
}

var photo = CreateImage();
if(photo==null)
if (photo == null)
throw new Exception("Couldn't create temporary Image");

var intent = CreateIntent(photo);
if(intent==null)
if (intent == null)
throw new Exception("Couldn't create Image capture intent");

ActivityUtils.StartActivity(intent, new TakePictureCallback(_promise).OnActivityResult, (object)photo);
Expand All @@ -75,9 +90,8 @@ namespace Fuse.Camera
try {
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

//FileProvider way for Marshmallow+ (API 23)
// FileProvider way for Marshmallow+ (API 23)
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.LOLLIPOP) {

File photoFile = p.getFile();
if (photoFile != null) {
Uri photoURI = FileProvider.getUriForFile(
Expand Down Expand Up @@ -153,7 +167,7 @@ namespace Fuse.Camera
_p.Reject(new Exception(reason));
}
}

[ForeignInclude(Language.Java, "android.provider.MediaStore", "com.fuse.Activity", "android.content.Intent", "com.fuse.camera.Image", "com.fuse.camera.ImageStorageTools", "androidx.core.content.ContextCompat")]
extern (Android) class CheckPermissionsCommand
{
Expand All @@ -162,29 +176,55 @@ namespace Fuse.Camera
var cb = new PromiseCallback<string>(p);
CheckPermissionsInternal(cb.Resolve, cb.Reject);
}

[Foreign(Language.Java)]
internal static void CheckPermissionsInternal(Action<string> onComplete, Action<string> onFail)
@{
if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.READ_EXTERNAL_STORAGE) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to read");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.WRITE_EXTERNAL_STORAGE) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to write");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.CAMERA) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
if (android.os.Build.VERSION.SDK_INT >= 33)
{
onFail.run("User does not have permission access the camera");
if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.READ_MEDIA_IMAGES) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to read images");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.READ_MEDIA_VIDEO) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to read videos");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.READ_MEDIA_AUDIO) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to read audio");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.CAMERA) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission access the camera");
}
else
{
onComplete.run("User has permission to read, write and access camera");
}
}
else
{
onComplete.run("User has permission to read, write and access camera");
if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.READ_EXTERNAL_STORAGE) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to read");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.WRITE_EXTERNAL_STORAGE) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission to write");
}
else if (ContextCompat.checkSelfPermission(com.fuse.Activity.getRootActivity(), android.Manifest.permission.CAMERA) != com.fuse.Activity.getRootActivity().getPackageManager().PERMISSION_GRANTED)
{
onFail.run("User does not have permission access the camera");
}
else
{
onComplete.run("User has permission to read, write and access camera");
}
}
@}
}

[ForeignInclude(Language.Java, "android.provider.MediaStore", "com.fuse.Activity", "android.content.Intent", "com.fuse.camera.Image", "com.fuse.camera.ImageStorageTools")]
extern (Android) class requestAndroidPermissions
{
Expand All @@ -193,21 +233,51 @@ namespace Fuse.Camera
{
_callback = new PromiseCallback<string>(p);
}

public void Execute()
{
Permissions.Request(new PlatformPermission[] { Permissions.Android.WRITE_EXTERNAL_STORAGE, Permissions.Android.READ_EXTERNAL_STORAGE, Permissions.Android.CAMERA }).Then(OnPermissions, OnRejected);
if (AndroidProperties.BuildVersion >= 33)
{
Permissions.Request(new PlatformPermission[] {
Permissions.Android.READ_MEDIA_IMAGES,
Permissions.Android.READ_MEDIA_VIDEO,
Permissions.Android.READ_MEDIA_AUDIO,
Permissions.Android.CAMERA
}).Then(OnPermissions, OnRejected);
}
else
{
Permissions.Request(new PlatformPermission[] {
Permissions.Android.WRITE_EXTERNAL_STORAGE,
Permissions.Android.READ_EXTERNAL_STORAGE,
Permissions.Android.CAMERA
}).Then(OnPermissions, OnRejected);
}
}

void OnPermissions(PlatformPermission[] grantedPermissions)
{
if(grantedPermissions.Length == 3)
if (AndroidProperties.BuildVersion >= 33)
{
_callback.Resolve("Success");
if (grantedPermissions.Length == 4)
{
_callback.Resolve("Success");
}
else
{
_callback.Reject("Required permissions were not granted");
}
}
else
{
_callback.Reject("Required permission was not granted.");
if (grantedPermissions.Length == 3)
{
_callback.Resolve("Success");
}
else
{
_callback.Reject("Required permissions were not granted");
}
}
}

Expand Down

0 comments on commit c00910c

Please sign in to comment.