Shortcomings of floating windows on Android

Václav HodekVáclav Hodek
9 min read

Have you ever wondered how to make those floating windows used by Facebook Heads and other apps? Have you ever wanted to use the same technology in your app? It’s easy, and I will guide you through the whole process.

I'm the author of Floating Apps; the first app of its kind on Google Play and the most popular one with over 8 million downloads. After 6 years of the development of the app, I know a bit about it. It’s sometimes tricky, and I spent months reading documentation and Android source code and experimenting. I received feedback from tens of thousands of users and see various issues on different phones with different Android versions.

Here's what I learned along the way.

Before reading this article, it's recommended to go through Floating Windows on Android 8: The Final App.

In this article, I teach you how to overcome some key shortcomings of floating technology.

Mixed Support

Below, I write about the most significant shortcomings of floating technology. On some devices, some of the described problems may not appear at all. On other devices, some issues exist, some not.

However, if you want your app to be available to all users, you can't rely on mixed support. Your app would crash frequently, or it wouldn't be usable for a large group of users.

Believe me, I had the privilege to see my app running on thousands of different devices, and I came across weird and illogical bugs and non-standard behavior.

Dialogs

Problem:

The standard Android's AlertDialog needs an underlying activity, and so it's not possible to use it in floating windows - it would cause the app to crash.

Solution:

The best solution is to implement custom dialogs based on floating technology.

In Floating Apps, I implemented many different dialog types:

  • yes/no
  • ok/cancel
  • ok only
  • select file
  • show list
  • list of checkboxes
  • rich-text
  • custom view
  • etc.

I also implemented modality. When the dialog is shown, the related window is blocked, and the dialog is brought to the front instead of the window.

Example:

A simple confirmation dialog.

Dialog in Floating Apps

Problem:

With popup menus, there is a similar problem as for AlertDialog. It's not supported in floating windows.

Solution:

As for AlertDialog, I would recommend implementing custom popup menus based on floating technology.

I did for Floating Apps. It relies on popups heavily.

Example:

A simple popup menu shown over its parent window.

Popup menu in Floating Apps

Copy & Paste

Problem:

The copy/paste feature is a real problem. It works well only on a few phones. As a rule of thumb, it's better to consider it unsupported.

Usually, it's possible to highlight/select text. To be more precise, it's possible to select words with a long tap. However, handles used for working with the selection are missing, and therefore, there is no way to select anything else than single words.

And there's another issue. The standard popup with actions (copy, paste, select all, translate, etc.) is somehow connected to the underlying activity on most devices and won’t appear at all.

Solution:

For short texts, it's acceptable to implement OnLongClickListener and use custom popup. However, as it's not possible to select the text this way, it's necessary to introduce Copy all/Cut all functionality. The Facebook Heads uses this solution everywhere. In Floating Apps, I use it too for edit fields.

In Floating Apps, there are mini-apps for notes taking and text editing, and being able to use Copy all only, it wouldn't be a suitable solution. For this reason, these apps use WebView in which a full-size contenteditable div is shown with custom logic for text selection, copying and pasting. It's not perfect, but better than nothing :-).

Example:

A custom copy/paste popup menu. Copy/paste in Floating Apps

Permissions

Problem:

Obtaining permissions from Android M and above requires a dialog to be shown and the method for issuing the request is available in Activity as well as the mechanism for catching the result.

Solution:

In fact, there are two possible solutions to this problem.

  1. You can ask the user to grant all the necessary permissions in the main app before you allow her to access floating windows. Honestly, this solution may scare your users, and as Floating Apps has around 8 required permissions, I believe that almost nobody would grant them all before any real experience with the app.
  2. Ask the user for the required permission when you need it. This is the correct way how to do it. In this case, the best you can do is to show an invisible activity and raise the permission request dialog from it. It may interrupt the current task, but as this is an only one-time action, it's not a big deal, and there is no better way to do.

In Floating Apps, every single mini-app defines a set of required permissions, and before the app is launched, the service checks whether all permissions are granted. If not, an invisible activity is shown and ask the user to grant missing ones.

System Interactions

Problem:

Some things, such as social logins, requesting SD card access, opening with, sharing, etc., can only be invoked from running activity.

Solution:

I tend to move "configuration" level things to the main app, which is a normal Android app, and once configured, it's usually possible to use them from inside the service.

For Open With / Share With functionality, a hidden activity is a solution both for showing the dialog to open/share with and for receiving data from other apps.

However, you may need to sacrifice functionality that is not supported in the service. For example, I once wanted to implement floating navigation with MapBox, but their SDK expects an instance of Activity to be available. No luck.

Z-Order

Problem:

The most recently added View is rendered on top of all others. There is no mechanism for switching the z-order of Views in WindowManager.

Solution:

In Floating Apps, the switching of floating windows is a complex process.

  • The whole window is rendered to an image, and a new ImageView is injected to WindowManager with the image of the original window on exactly the same position. In fact, it looks exactly the same as the original window.
  • The original window is removed from the WindowManager but kept alive.
  • The original window is added back to the WindowManager, which means that it's effectively brought to the top.
  • The ImageView with the rendered image of the window is removed from the WindowManager.

Using the approach above, switching windows seems like a seamless process without blinking and other undesired effects.

Of course, in Floating Apps, this is automated, and there is a complex logic around this that also controls the current z-order of all windows and intelligently decides when the switching is necessary and when not.

However, removing and re-adding the window may cause a problem with components rendering content directly to the video memory such as SurfaceView, VideoView, etc. For this reason, Floating Apps automatically disable this process for windows of a certain type.

Browser Limitations

WebView works well in floating windows, and it's mostly possible to reimplement its interactions such as uploading and downloading files, invalid certificate notifications, requests for location permission, etc. to use floating windows/dialogs described above.

However, there are a few actions that may lead to crashes or are unsupported. I hope I listed all the key issues.

Open With

Android tries to offer available apps, aka Open With feature, when opening links with an unknown protocol. As there is no activity, it may cause crashes. However, this behavior can be easily overridden.

Copy/Paste

The problem with copy/paste described above also exists for WebView, and it's even worse.

It's possible to use WebView to implement a workaround for unsupported text selection because we have full control over the code. However, for other websites, the situation is much complicated.

However, it's possible to get the selected text in the WebView using something like:

webView.loadUrl("javascript:process(window.getSelection().toString());")

And paste the text to WebView using simulating key strokes with dispatchKeyEvent.

Long-clicks may lead to crashes depending on the implementation, and certainly, it's not possible to invoke standard actions like Open In New Window, Download Image, etc.

The solution is to set custom OnLongClickListener for the WebView and get use webView.getHitTestResult() to get information about the clicked element.

Don't forget to return false when webView.getHitTestResult() returns null to let the WebView process the long click correctly, e.g. for selecting texts.

HTML <select>

HTML element <select> shows a dialog with available options and cause your app to crash. It's a big deal, and the only solution is to hook the element, override its action, and show custom UI based on floating technology.

It's not a simple task and needs a lot of testing. Don't forget that options in <select> may be nested and disabled.

The Final Note

You learned how to implement floating windows for your app. It's a powerful technology, but it comes with its price.

Use it wisely where it makes sense and try to avoid situations in which you get to shortcomings mentioned in this article and some others I forgot. It's not a simple task to correctly implement all the workarounds to be reliable across different phones and Android versions.

Enjoy it and feel free to share your apps with me.

Also, I will be glad if you decide to localize your apps with Localazy. I put the very same love into it as I did with Floating Apps.

Continue to the last article with tips & tricks I learned the hard way.

The Series

This article is part of the Floating Windows on Android series.

28
Subscribe to my newsletter

Read articles from Václav Hodek directly inside your inbox. Subscribe to the newsletter, and don't miss out.

Written by

Václav Hodek
Václav Hodek