如何在android系统状态栏中显示文本
Posted
技术标签:
【中文标题】如何在android系统状态栏中显示文本【英文标题】:How do I show text in android system status bar 【发布时间】:2018-01-04 16:32:18 【问题描述】:我正在尝试为 android Nougat 开发一个应用程序,我想在状态栏中显示一些从 android 服务例程生成的信息/文本。所以我的问题是,我不知道如何在状态栏中显示文本。
我添加了一张示例图片来说明我的意思(红色圆圈)。 我知道这是可能的,因为我在 Play 商店的电池监控应用中看到了它。
我已经尝试过 NotificationCombat.Builder,但我认为这不是正确的方法。也许是叠加层,但搜索后我没有找到任何东西。
谁能告诉我怎么做或给我一个提示,好吗?
编辑:这是我的 NotificationCompat.Builder 测试代码
MainActivity.java
import android.app.Notification;
import android.app.NotificationManager;
import android.os.Bundle;
import android.support.v4.app.NotificationCompat;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
public class MainActivity extends AppCompatActivity
private final int NOTIFICATION_ID = 10;
@Override
protected void onCreate (Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(this);
mBuilder.setContentTitle("Value");
mBuilder.setContentText("123");
mBuilder.setSmallIcon(R.mipmap.ic_launcher);
mBuilder.setOngoing(true);
mBuilder.setAutoCancel(false);
//Intent resultIntent = new Intent(this, MainActivity.class);
//PendingIntent resultPendingIntent = PendingIntent.getActivity(this, 0, resultIntent, PendingIntent.FLAG_UPDATE_CURRENT);
//mBuilder.setContentIntent(resultPendingIntent);
Notification notification = mBuilder.build();
notification.flags |= Notification.FLAG_NO_CLEAR | Notification.FLAG_ONGOING_EVENT;
NotificationManager mNotifyMgr = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
mNotifyMgr.notify(NOTIFICATION_ID, notification);
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_
android:layout_
android:fitsSystemWindows="true"
tools:context=".MainActivity">
<android.support.design.widget.CoordinatorLayout
android:layout_
android:layout_>
<android.support.design.widget.AppBarLayout android:layout_
android:layout_
android:theme="@style/AppTheme.AppBarOverlay">
<android.support.v7.widget.Toolbar android:id="@+id/toolbar"
android:layout_
android:layout_
android:background="#000000"
app:popupTheme="@style/AppTheme.PopupOverlay" />
</android.support.design.widget.AppBarLayout>
<LinearLayout
android:layout_
android:layout_
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true"
android:orientation="vertical"
app:layout_behavior="@string/appbar_scrolling_view_behavior"
android:weightSum="100" >
<TextView
android:id="@+id/tv_value"
android:layout_
android:layout_
android:text="Hello World!"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"/>
</LinearLayout>
</android.support.design.widget.CoordinatorLayout>
</RelativeLayout>
结果:
【问题讨论】:
Display Notification Text in Status Bar - Android 的可能副本希望,这就是你要找的...... 我已经阅读并尝试过,但这不是我要搜索的内容。我不需要通知即可向下滑动,我需要直接在状态栏中显示文本。我尝试使用 NotificationCompat.Builder 中的 setSmallIcon(int icon),但它需要一个 int 资源,因此无法将文本用于位图/图像。 @Matze 你的问题解决了吗? 【参考方案1】:我确实找到了解决方案,关键字是 overlay 和 floating window。
int statusBarHeight = 0;
int resourceId = getResources().getIdentifier("status_bar_height", "dimen", "android");
if (resourceId > 0) statusBarHeight = getResources().getDimensionPixelSize(resourceId);
final WindowManager.LayoutParams parameters = new WindowManager.LayoutParams(
WindowManager.LayoutParams.WRAP_CONTENT,
statusBarHeight,
WindowManager.LayoutParams.TYPE_SYSTEM_ERROR, // Allows the view to be on top of the StatusBar
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN, // Keeps the button presses from going to the background window and Draws over status bar
PixelFormat.TRANSLUCENT);
parameters.gravity = Gravity.TOP | Gravity.CENTER;
LinearLayout ll = new LinearLayout(this);
ll.setBackgroundColor(Color.TRANSPARENT);
LinearLayout.LayoutParams layoutParameteres = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.MATCH_PARENT);
ll.setLayoutParams(layoutParameteres);
TextView tv = new TextView(this);
ViewGroup.LayoutParams tvParameters = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.MATCH_PARENT);
tv.setLayoutParams(tvParameters);
tv.setTextColor(Color.WHITE);
tv.setGravity(Gravity.CENTER);
tv.setText("123");
ll.addView(tv);
WindowManager windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
windowManager.addView(ll, parameters);
【讨论】:
我收到以下错误 android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl-- permission denied for window type 2010 .. 知道为什么或如何解决它吗? 错误:窗口类型 2010 的权限被拒绝【参考方案2】:嗯,我做到了。我使用了将text
转换为图标,然后在状态栏上显示的方式。一些成员试图覆盖android不允许的状态栏(SDK> = 22),我不知道这是否适用于某人。但是将text
转换为图标对我来说非常有效。 在 Oreo
上测试过
伪代码
-
将文本转换为
Bitmap
然后将Bitmap
转换为图标。
在状态栏上显示该图标。
输出
代码如下:
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
displayNotification("5F");
public void displayNotification(String text)
Notification.Builder builder = null;
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O)
builder = new Notification.Builder(this, CHANNEL_ID);
//convert text to bitmap
Bitmap bitmap = createBitmapFromString(text.trim());
//setting bitmap to staus bar icon.
builder.setSmallIcon(Icon.createWithBitmap(bitmap));
builder.setContentTitle("Simple Notification");
builder.setContentText("This is a simple notification");
builder.setPriority(Notification.PRIORITY_MAX);
NotificationManagerCompat notificationManagerCompat = NotificationManagerCompat.from(this);
notificationManagerCompat.notify(NOTIFICATION_ID, builder.build());
createNotificationChannel();
private void createNotificationChannel()
// Create the NotificationChannel, but only on API 26+ because
// the NotificationChannel class is new and not in the support library
if (VERSION.SDK_INT >= VERSION_CODES.O)
CharSequence name = "testing";
String description = "i'm testing this notification";
int importance = NotificationManager.IMPORTANCE_DEFAULT;
NotificationChannel channel = new NotificationChannel(CHANNEL_ID, name, importance);
channel.setDescription(description);
// Register the channel with the system; you can't change the importance
// or other notification behaviors after this
NotificationManager notificationManager = getSystemService(NotificationManager.class);
assert notificationManager != null;
notificationManager.createNotificationChannel(channel);
private Bitmap createBitmapFromString(String inputNumber)
Paint paint = new Paint();
paint.setAntiAlias(true);
paint.setTextSize(100);
paint.setTextAlign(Paint.Align.CENTER);
Rect textBounds = new Rect();
paint.getTextBounds(inputNumber, 0, inputNumber.length(), textBounds);
Bitmap bitmap = Bitmap.createBitmap(textBounds.width() + 10, 90,
Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
canvas.drawText(inputNumber, textBounds.width() / 2 + 5, 70, paint);
return bitmap;
如果有人知道更好的方法,请提及
【讨论】:
好主意。我已经实现了这个及其工作。但唯一的问题是我收到了很多糟糕的通知崩溃。我搜索了一下,发现 builder.setSmallIcon() 参数应该是 PNG。您知道如何将位图转换为 PNG 并将其传递给 builder.setSmallIcon() 吗?【参考方案3】:你可以在文档中找到你的答案,这里:https://developer.android.com/reference/android/support/v4/app/NotificationCompat.html
编辑:: 好吧,答案在文档中。然而,经过大量的研究和挖掘,似乎社区之间的共识是,这对于任何应用程序都是不可能的。只有特定图标可以放置在状态栏的右侧(即时钟、天气、系统信息等...)。
很抱歉,没有比这更令人兴奋的答案了,但至少你可以不用再为自己无法弄清楚的原因而烦恼了。
编辑 2:: 显然,棒棒糖之前的设备可以访问允许您使用系统图标的私有 api(再次考虑一下警报图标)。之后,api被删除。这个*** post 相当广泛地讨论了整个情况。
编辑 3:: 如果您可以忍受将图标放在状态栏的左侧,您可以将文本转换为位图,如下所示:
TextView textView = new TextView(activity.getContext());
textView.setText("Hello World");
textView.setDrawingCacheEnabled(true);
textView.destroyDrawingCache();
textView.buildDrawingCache();
Bitmap bitmap = getTransparentBitmapCopy(textView.getDrawingCache());
private Bitmap getTransparentBitmapCopy(Bitmap source)
int width = source.getWidth();
int height = source.getHeight();
Bitmap copy = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
int[] pixels = new int[width * height];
source.getPixels(pixels, 0, width, 0, 0, width, height);
copy.setPixels(pixels, 0, width, 0, 0, width, height);
return copy;
【讨论】:
你能说得具体一点吗?我尝试使用 NotificationCompat.Builder,但它没有按需要工作(上图示例)。 请随时发布您的代码和结果图片。 我编辑了我的帖子,所以你可以看到我的代码。请告诉我我错过了文档的哪一部分。 我怎么说,NotificationCompat 不是正确的方式,但有一种工作方式,在应用程序电池监视器(GLGJing)中演示。也许有人可以使用覆盖布局来做到这一点。如果这是正确的方法,我现在不知道,但是 ActionBarOverlayLayout 有可能吗?我读过一些关于它的东西,但并不多,而且我在 android 文档中没有找到真正有用的信息。 您正在寻找的功能是不可能的。我将编辑我的答案以进一步澄清。以上是关于如何在android系统状态栏中显示文本的主要内容,如果未能解决你的问题,请参考以下文章
如何使用 VBA 从 Outlook 状态栏中删除 DASL 过滤器文本?