Skip to content
This repository was archived by the owner on May 15, 2024. It is now read-only.

Launcher API #250

Closed
adamfisher opened this issue May 12, 2018 · 19 comments
Closed

Launcher API #250

adamfisher opened this issue May 12, 2018 · 19 comments

Comments

@adamfisher
Copy link

adamfisher commented May 12, 2018

Description of the Feature

Provide the ability to detect and launch another application already installed on the device.

//Updated by JM

API Declaration

Integration Point: Launcher

API:

- Task<bool> CanOpenAsync(string uri);
- Task<bool> CanOpenAsync(Uri uri);
- Task OpenAsync(string uri);
- Task OpenAsync(Uri uri);

Implementations:

CanOpenAsync:

OpenAsync:

  • UWP: Launcher LaunchUriAsync
  • Android: Standard launcher stuff
  • iOS: UIApplication.SharedApplication.OpenURL

Documentation links for supported platforms

@adamfisher
Copy link
Author

adamfisher commented May 12, 2018

I realize there is some overlap with the way the Browser API works due to it handling HTTP protocols so this proposal could morph into enhancing it to support launching other applications. iOS specifically appears to be using the same code that would open another application when using BrowserLaunchType.External.

Changing the Browser API to Launcher API seems like a more appropriate generic/high-level thing since launching the device browser is only one such app. What do you guys think?

@jamesmontemagno
Copy link
Collaborator

I am pretty open to changing the name here to Launcher instead of Browser as this should work out of the box with our current implementation. New check I think is good for this instance as well. Will triage Monday.

@jamesmontemagno
Copy link
Collaborator

The question to me will be is it alright to have our single API for opening uri.. or perhaps just a new API that is always extrenal. Leaning that way.

@mattleibow
Copy link
Contributor

Yeah, we want the ability for the browser to be more specialized. It needs to support Chorme/Safari views.

I am not such a fan of merging the two concepts.

@jamesmontemagno jamesmontemagno added proposal Proposed feature or enhancement. needs-specifications Accepted feature/enhancement and needs specification. labels May 14, 2018
@jamesmontemagno
Copy link
Collaborator

jamesmontemagno commented May 14, 2018

Agreed it should be in the "Launcher" name.

I will be updating the main comment now with discussion points.

@Redth
Copy link
Member

Redth commented May 14, 2018

On Android we should essentially use QueryIntentActivities to see if there are any activities available to match our intent (which likely is just created with ACTION_VIEW).

@jamesmontemagno
Copy link
Collaborator

The only question i have is if we need the application id at all or just the uri is fine.

The only thing would be to launch a specific app if there are multiple that supports the scheme... not sure if that is supported on all platforms (assumption is no).

@Redth
Copy link
Member

Redth commented May 14, 2018

I think taking the app id into consideration makes things considerably more challenging or impossible on some platforms (eg: iOS).

In practice I think most apps declare their own unique uri scheme anyway.

@jamesmontemagno
Copy link
Collaborator

yeah, i would leave it off for now in first iteration.

@jamesmontemagno jamesmontemagno added ready-to-implement Feature approved, specs written, and ready to implement. and removed needs-specifications Accepted feature/enhancement and needs specification. proposal Proposed feature or enhancement. labels May 14, 2018
@jamesmontemagno jamesmontemagno added this to the 0.7.0-preview milestone May 14, 2018
@adamfisher
Copy link
Author

Since Android provides a non-URI option, it will not be possible to cover it since we are going for lowest common denominator of the API surface area but you can open an application using it's package name:

Intent launchIntent = getPackageManager().getLaunchIntentForPackage("com.example.abc");
startActivity( launchIntent );

If the proposed methods will only be used to open other apps, what do you think about some convenience methods to open a specific app store page? Identifier would be the package name on Android and the appId on iOS. Might be too much since you could already do it with the OpenAsync(...) methods that would be available.

Something like:

- Task OpenAppStoreListingAsync(string identifier);

Arguably, any API that invokes an external application could be a candidate for the Launcher API
(replacing Sms, PhoneDialer, Email APIs) since launcher is just a higher-level abstraction.

So the complete API could be like this:

- Task<bool> CanOpenAsync(string uri);
- Task<bool> CanOpenAsync(Uri uri);
- Task OpenAsync(string uri);
- Task OpenAsync(Uri uri);
- Task OpenAppStoreListingAsync(string identifier);
- bool OpenDialer(string phoneNumber); // true - dialer opened successfully
- Task ComposeEmailAsync(EmailMessage message);
- Task ComposeSmsAsync(SmsMessage message);

That might be going a little too crazy but fun to brainstorm 😃

@mattleibow
Copy link
Contributor

We probably won't merge in the sms, email, dialer because we probably will extend the APIs. For example, the dialer my get an actual start call, or the sms a background send.

But, you are right when saying that email (as it is today) can just use the launcher API internally. We actually do use our own apis in other apis here.

@jamesmontemagno
Copy link
Collaborator

Yeah, ideally that route. I think the API I outlined in the original post is the go to.

@Mrnikbobjeff
Copy link
Contributor

As far as I understood the contribution guide I may now implement this feature? I began working on it on https://github.com/Mrnikbobjeff/Essentials/tree/feature/issue-250. I implemented it for all Platforms, though I only ran unit tests on the Android implementation. If it were possible I'd like to be assigned this issue so that it could be marked as "In Progress".

@Redth Redth added the in-progress Actively being worked on. label Jul 3, 2018
@Redth
Copy link
Member

Redth commented Jul 3, 2018

Thanks @Mrnikbobjeff I've marked it as in progress :)

@jamesmontemagno jamesmontemagno added the up-for-grabs Implementation from community can be started. label Jul 3, 2018
@Mrnikbobjeff
Copy link
Contributor

Mrnikbobjeff commented Jul 4, 2018

@Redth @jamesmontemagno I implemented the UWP and Android Launcher and tested them. I would like to have clarified what should happen if the user inputs a string which is not a valid System.Uri. On UWP this would throw a InvalidOperationException, on Android this would simply return false on CanOpenAsync. Should we enforce that all URI's are valid or are we to accept different behviour on different platforms?
Check Can_Not_Open_Uri_As_String_Throws_InvalidUriException for different behaviour on different platforms.

As I see it we have three options:
1.) Try/Catch and return false, thus hiding the invalid uri exception.
2.) Enforce Uri validity, thus throwing on every plaftorm.
3.) Different exception behaviour on different platforms.

@Mrnikbobjeff
Copy link
Contributor

I went with option 3 for the moment, UWP and iOS throw UriFormatExceptions, Android simply returns false. Any other input on this matter?

@Redth
Copy link
Member

Redth commented Jul 6, 2018

I personally like option 2.

We generally want to keep things consistent between platforms whenever possible, and this is a relatively easy way to do so.

@Mrnikbobjeff
Copy link
Contributor

Okay, I changed it to verify the parameter on every platform. Once I return to a pc running windows I will update the documentation and create a PR

jamesmontemagno pushed a commit that referenced this issue Jul 23, 2018
* Initial commit for Launcher Api as discussed in Issue 250

* Fixed and refactored android launch code, fixed unit tests. Fixed namespaces.

* Refactored code, fixed uwp launcher. Added unit test

* Added sample launcher page and viewmodel.

* Fixed launcher Page typo

* Refactored launcher code. after testing exception behaviour I decided on rethrowing instead of swallowing the exception

* Refactored access modifiers. Also changed exception behaviour to enforce uri validity and exception behaviour across platforms.

* Updated docs, stripped even more code.

* Added validation. Adressed all other changes as required in pr

* Removed dead code

* Adressed last issues from pr
Redth pushed a commit that referenced this issue Jul 25, 2018
* GH-250 Implement Launcher API (#356)

* Initial commit for Launcher Api as discussed in Issue 250

* Fixed and refactored android launch code, fixed unit tests. Fixed namespaces.

* Refactored code, fixed uwp launcher. Added unit test

* Added sample launcher page and viewmodel.

* Fixed launcher Page typo

* Refactored launcher code. after testing exception behaviour I decided on rethrowing instead of swallowing the exception

* Refactored access modifiers. Also changed exception behaviour to enforce uri validity and exception behaviour across platforms.

* Updated docs, stripped even more code.

* Added validation. Adressed all other changes as required in pr

* Removed dead code

* Adressed last issues from pr

* Cleanup launcher

* cleanup iOS tests

* Early morning typos
@Redth Redth removed in-progress Actively being worked on. ready-to-implement Feature approved, specs written, and ready to implement. up-for-grabs Implementation from community can be started. labels Jul 25, 2018
@Redth
Copy link
Member

Redth commented Jul 25, 2018

Merged.

@Redth Redth closed this as completed Jul 25, 2018
jamesmontemagno added a commit that referenced this issue Jul 31, 2018
* GH-343: Take into consideration VerticalAccuracy on iOS Altitude (#344)

* Fix for #343

Only set altitude if it's available (VerticalAccuracy is negative when it isn't according to iOS doc)

* Use: (double?)

* dotnet foundation CLA broken link (#349)

* Update ns-Xamarin.Essentials.xml

* Update README.md (#340)

* GH-365 Change to generic EventHandlers

* Clean up unit tests.

* GH-363 Enable C# 7.2 (#364)

* GH-363 Enable CD 7.3

* Update Xamarin.Essentials.csproj

* GH-367 Change Ui to UI & Ac to AC (#374)

* GH-367 Change Ui to UI

* Ensure proper Ui for Android

* Fix gryoscope

* Rename Ac to AC

* Update framework indexes

* Create Product Feedback page for Docs (#369)

* Create PRODUCT-FEEDBACK.md

* Update PRODUCT-FEEDBACK.md

* Fixed a typo from "chanaged" to "changed"

* Fixed a typo from "chanaged" to "changed"

* Fixed a typo from "chanaged" to "changed"

* GH-337 Android Low Pass Filter (#354)

* Implemented low pass filter as requested by issue 337

* Added value to sample page.

* Updated docs for ApplyLowPassFilter

* Enabled ApplyLowPassFilter Switch on Android, disabled on all other platforms

* Moved LowPassFilter to shared, removed unnecessary usings.

* Updated Docs

* Sample Project Tweaks and Optimizations (#353)

* Make sure orientation sensor stops on sample.

* Fix up all sensors page to start and stop correctly

* Only check permission if unknown status

* Update to latest sdk.extras

* fix back button press on TTS page. Check for null.

* Add samples sln for CI/CD

* Update readme and samples sln

* Don't use code sign key for simbuilds

* let's try this gain

* Update to projects

* Cleanup sample config

* bump sdk extras

* Add in readme for install with helpful information. Bump MSBuild.Sdk.Extras (#392)

* GH-368: Allow for unicode in host names and escape query (#393)

* Allow for unicode in host names and escape query

Previously we were passing the the url as is to each platform and letting it deal with how to parse into the platform specific url.

This handles unicode domain names (IDN Mapping aka PunyCode) for the domain, but also reconstructs the url based on the parsed System.Uri from the original string, and so the PathAndQuery will be the % escaped version which will account for unicode characters in it as well.

Finally, on iOS and Android we are using AbsoluteUri to end up with the fully IDN mapped and % encoded URL to pass to the system URL types.

* Extract Escape Uri logic into its own method

More testable.

* Add uri escaping tests

* Improve uri escape tests

* Update default iOS DeviceTests target

Also don’t allow picking a sim that’s unavailable (this was a bug since we were checking for contains `available` which of course `unavailable` also contains!)

* Update iOS DeviceInfo.Model to return specifid hw.machine for more control. (#395)

via:
https://github.com/xamarin/xamarin-macios/blob/bc492585d137d8c3d3a2ffc827db3cdaae3cc869/tests/linker/ios/link%20sdk/LinkSdkRegressionTest.cs#L603-L631

Added tests

* GH-380 Use Internal Preferences for Secure Storage consistency. (#386)

* Use Internal Preferences for Secure Storage consistency.
Save if we created key pre-M so we always use pre-M if device upgrades.

* Fix logic for pre-m key check

The logic was slightly off, I think this fixes it, but would be good to have another set of eyes...

1. We check to see if the device is pre-M (if it does _not_ have `M`, or if we already set the fact it's pre-M in the preference - aka from a previous install before an upgrade of the OS)
2. If we aren't pre-M, we can't use Symmetric Key from Keystore
3. If we make it down to using Asymmetric Key, we set the pre-M preference to `true` to persist the value for future invocations, which will make it 'stick' in the event of a pre-M to M+ OS upgrade.

* Address feedback on key naming.

* Added test for secure storage to simulate upgrade

From API < 23 to API >= 23 after storing data with an asymmetric key and then moving to a platform supporting symmetric keys.

* Ensure we always set flags when using specified keygen

* GH-388 If activity is required and null throw null exception with information. (#396)

* If activity is required and null throw null exception with information.

* Remove currentactivity and update getcurrentactivity.

* GH-250 Implement Launcher API (#356) (#405)

* GH-250 Implement Launcher API (#356)

* Initial commit for Launcher Api as discussed in Issue 250

* Fixed and refactored android launch code, fixed unit tests. Fixed namespaces.

* Refactored code, fixed uwp launcher. Added unit test

* Added sample launcher page and viewmodel.

* Fixed launcher Page typo

* Refactored launcher code. after testing exception behaviour I decided on rethrowing instead of swallowing the exception

* Refactored access modifiers. Also changed exception behaviour to enforce uri validity and exception behaviour across platforms.

* Updated docs, stripped even more code.

* Added validation. Adressed all other changes as required in pr

* Removed dead code

* Adressed last issues from pr

* Cleanup launcher

* cleanup iOS tests

* Early morning typos

* Name Alignment for BrowserLaunchMode (#408)

* Rename BrowserLaunchType to BrowserLaunchMode

* update nuget

* Rename parameter

* GH-287 OpenMaps Implementation (#361) (#404)

* GH-287 Maps Implementation (#361)

* Implemented maps as mentioned in branch name issue. also fixed a spelling mistake

* Added samples

* Added doc stubs

* Added comments for map types

* Refactored code

* Formatting of docs

* Added tests

* Changed coordinates to display something recognisable

* Added uri escaping, thus removing duplicate code. also had to turn off a warning due to inconsistent style cop and vs warning behaviour.

* Removed ref until in is added in c# 7.2

* Adressed issues in pr

* Updated launch code

* Adressed method to onliner

* Updated docs

* Removed ClearTop intent, added sample with adress

* Rename to align names. Added extensions and additional paramaters.

* Update tests

* Throw if options are null.

* Add overload for lat/long without options

* Enable multi-target for 25,26,27 (#411)

Alight nuget versions to xamarin.forms.

* No More Shared Projects (#410)

* Change to multi-targeted project for device tests :)

* Cleanup scripts

* Fix build error due to preprocessor directive (#417)

* Sync Sensor Speeds & Reset UWP to 0 (#419)

* Restore the default report interval to release resources while the sensor is not in use

* Cleanup sensor reading times.
Make them all the same and only in 1 place
Compass is still unique

* Create unique preference storage for each feature. (#434)

* Create unique preference storage for each feature.

* update storage per @Redth

* Update version
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

No branches or pull requests

5 participants