Floating Bubble View
https://github.com/dofire/Floating-Bubble-View
π Library Overview
An Android library that creates floating bubbles on top of the screen π¨, supports both XML and π Jetpack Compose
πFloating Bubble View
An Android library that creates floating bubbles on top of the screen π¨, supports both XML and π Jetpack Compose
Like this project? π₯° Don't forget to show some love by giving a Starβ
| Bubble | Custom | | :-: | :-: | |

Β
Variants
-
Flutter
If you are looking for a Flutter version of this library, check dash_bubble, a Flutter plugin that allows you to create a floating bubble on the screen. by Moaz El-sawaf.
Table of Contents π
I, Getting started πππ
Ensure your appβs minimum SDK version is 21+ and `mavenCentral()` included
</br> 1. Ensure your appβs minimum SDK version is 21+. This is declared in the module-level `build.gradle` file ```gradle android { defaultConfig { ... minSdk 21 } ``` 2. Ensure the `mavenCentral()` repository is declared in the project-level `build.gradle`/`setting.gradle` file:settings.gradle
```gradle pluginManagement { repositories { ... mavenCentral() } } dependencyResolutionManagement { ... repositories { ... mavenCentral() } } ```build.gradle (project-level) (on old gradle versions)
```gradle allprojects { repositories { mavenCentral() ... } ... } ```Declare the dependencies in the module-level build.gradle
file π
dependencies {
implementation("io.github.torrydo:floating-bubble-view:<LATEST_VERSION>")
}
II, Setup πβπ°
1, extends ExpandableBubbleService()
and call expand()
or minimize()
1οΈβ£
Java
```java Java docs is not completed yet because the author is (really) busy and (a little) tired πͺ ```Kotlin
```kotlin class MyService: ExpandableBubbleService() { override fun onCreate() { super.onCreate() minimize() } // optional, only required if you want to call minimize() override fun configBubble(): BubbleBuilder? { return ... } // optional, only required if you want to call expand() override fun configExpandedBubble(): ExpandedBubbleBuilder? { return ... } } ```</br>
2, add bubble service to the manifest file 2οΈβ£
<application>
<!-- these two permissions are added by default -->
<!-- <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/> -->
<!-- <uses-permission android:name="android.permission.FOREGROUND_SERVICE"/> -->
<!-- You can find more permissions, use cases here: https://developer.android.com/about/versions/14/changes/fgs-types-required -->
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_SPECIAL_USE" />
<service
android:name="<YOUR_PACKAGE>.MyService"
android:foregroundServiceType="specialUse"
>
<property android:name="android.app.PROPERTY_SPECIAL_USE_FGS_SUBTYPE" android:value="foo"/> <!-- optional -->
</service>
</application>
Android 13 and earlier
```xml</br>
3, start bubble service and enjoy 3οΈβ£ ππ
Make sure βdisplay over other appsβ permission is granted, otherwise the app will crash β βπ₯
Java
```java Intent intent = new Intent(context, MyService.class); ContextCompat.startForegroundService(this, intent); // or // startService(intent); // for android version lower than 8.0 (android O) // startForegroundService(intent); // for android 8.0 and higher ```Kotlin
```kotlin val intent = Intent(context, MyService::class.java) ContextCompat.startForegroundService(this, intent) // or // startService(intent) // for android version lower than 8.0 (android O) // startForegroundService(intent) // for android 8.0 and higher ```</br>
III, Usage π₯
1, configBubble()
and configExpandedBubble()
Java
```java public class MyServiceJava extends ExpandableBubbleService { @Override public void onCreate() { super.onCreate(); minimize(); } @Nullable @Override public BubbleBuilder configBubble() { View imgView = ViewHelper.fromDrawable(this, R.drawable.ic_rounded_blue_diamond, 60, 60); imgView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { expand(); } }); return new BubbleBuilder(this) .bubbleView(imgView) .bubbleStyle(R.style.default_bubble_style) .bubbleDraggable(true) .forceDragging(true) .closeBubbleView(ViewHelper.fromDrawable(this, R.drawable.ic_close_bubble)) .closeBubbleStyle(R.style.default_close_bubble_style) .distanceToClose(100) .triggerClickablePerimeterPx(5f) .closeBehavior(CloseBubbleBehavior.FIXED_CLOSE_BUBBLE) .startLocation(100, 100) .enableAnimateToEdge(true) .bottomBackground(false) .addFloatingBubbleListener(new FloatingBubbleListener() { @Override public void onFingerDown(float x, float y) {} @Override public void onFingerUp(float x, float y) {} @Override public void onFingerMove(float x, float y) {} }) ; } @Nullable @Override public ExpandedBubbleBuilder configExpandedBubble() { View expandedView = LayoutInflater.from(this).inflate(R.layout.layout_view_test, null); expandedView.findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { minimize(); } }); return new ExpandedBubbleBuilder(this) .expandedView(expandedView) .startLocation(0, 0) .draggable(true) .style(R.style.default_bubble_style) .fillMaxWidth(true) .enableAnimateToEdge(true) .dimAmount(0.5f); } } ```Kotlin
```kotlin class MyServiceKt : ExpandableBubbleService() { override fun onCreate() { super.onCreate() minimize() } override fun configBubble(): BubbleBuilder? { val imgView = ViewHelper.fromDrawable(this, R.drawable.ic_rounded_blue_diamond, 60, 60) imgView.setOnClickListener { expand() } return BubbleBuilder(this) // set bubble view .bubbleView(imgView) // or our sweetie, Jetpack Compose .bubbleCompose { BubbleCompose() } // set style for the bubble, fade animation by default .bubbleStyle(null) // set start location for the bubble, (x=0, y=0) is the top-left .startLocation(100, 100) // in dp .startLocationPx(100, 100) // in px // enable auto animate bubble to the left/right side when release, true by default .enableAnimateToEdge(true) // set close-bubble view .closeBubbleView(ViewHelper.fromDrawable(this, R.drawable.ic_close_bubble, 60, 60)) // set style for close-bubble, null by default .closeBubbleStyle(null) // DYNAMIC_CLOSE_BUBBLE: close-bubble moving based on the bubble's location // FIXED_CLOSE_BUBBLE (default): bubble will automatically move to the close-bubble when it reaches the closable-area .closeBehavior(CloseBubbleBehavior.DYNAMIC_CLOSE_BUBBLE) // the more value (dp), the larger closeable-area .distanceToClose(100) // enable bottom background, false by default .bottomBackground(true) .addFloatingBubbleListener(object : FloatingBubbleListener { override fun onFingerMove(x: Float, y: Float) {} // The location of the finger on the screen which triggers the movement of the bubble. override fun onFingerUp(x: Float, y: Float) {} // ..., when finger release from bubble override fun onFingerDown(x: Float, y: Float) {} // ..., when finger tap the bubble }) // set the clickable perimeter of the bubble in pixels (default = 5f) .triggerClickablePerimeterPx(5f) } override fun configExpandedBubble(): ExpandedBubbleBuilder? { val expandedView = LayoutInflater.from(this).inflate(R.layout.layout_view_test, null) expandedView.findViewById### 2, Override default `Notification`
Java
```java public class MyService extends ExpandableBubbleService { ... @Override public void startNotificationForeground() { startForeground(...); // or you can use NotificationHelper class // val noti = NotificationHelper(this) // noti.createNotificationChannel() // startForeground(noti.notificationId, noti.defaultNotification()) } } ```Kotlin
```kotlin class MyService : FloatingBubbleService() { ... // optional, of course override fun startNotificationForeground() { startForeground(...) // or you can use NotificationHelper class // val noti = NotificationHelper(this) // noti.createNotificationChannel() // startForeground(noti.notificationId, noti.defaultNotification()) } } ```Notice since Android 13 β
Starting in Android 13 (API level 33), notifications are only visible if the "POST_NOTIFICATIONS" permission is granted.
> The service will run normally even if the notification is not visible. π > You still need to initialize the notification before showing any view.
### 3, Methods in `ExpandableBubbleService` | Name | Description | | :- | :- | | `removeAll()` | remove all bubbles | | `expand()` | show expanded-bubble | | `minimize()` | show bubble | | `enableBubbleDragging()` | enable bubble dragging or not | | `enableExpandedBubbleDragging()` | enable expanded-bubble dragging or not | | `animateBubbleToEdge()` | animate bubble to edge of the screen | | `animateExpandedBubbleToEdge()` | animate expanded-bubble to edge of the screen |
### 4, Helper Class - ViewHelper() - fromBitmap(context, bitmap): View - fromBitmap(context, bitmap, widthDp, heightDp): View - fromDrawable(context, drawableRes): View - fromDrawable(context, drawableRes, widthDp, heightDp) - NotificationHelper(context, channelId, channelName, notificationId) - notify(Notification): update notification based on notificationId - createNotificationChannel(): create notification channel from `android 8` and above - defaultNotification(): return default notification </br> ## IV, Contribution Guide π Contributions are welcome! π - If you come across a bug or have an idea for a new feature, please let us know by creating an [Issue](https://github.com/TorryDo/Floating-Bubble-View/issues) ππ‘ - If you're interested in taking on an [open issue](https://github.com/TorryDo/Floating-Bubble-View/issues), please comment on it so others are aware π - If you've already fixed a bug or implemented a feature, feel free to submit a [Pull request](https://github.com/TorryDo/Floating-Bubble-View/pulls) π - Having questions, ideas, or feedback? Don't worry, I gotchu. Simply open a [Discussion](https://github.com/TorryDo/Floating-Bubble-View/discussions) π - Find this project useful? π₯° Don't forget to show some love by giving a star β Thank you! π
## V, Work in Progress π§ This library is still under heavy development. There is still a lot of code cleanup to do, so expect breaking API changes over time. Please refer to the following page to check out the change-log: [Releases](https://github.com/TorryDo/Floating-Bubble-View/releases) Everything's gonna be ok! π
## VI, License ``` Copyright 2022 TorryDo Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ```