- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我们如何才能在 TextView
的最后一行实现淡出效果,就像在 Play 商店应用的“WHAT'S NEW”部分?
最佳答案
淡入淡出效果可以通过子类化 TextView
类来拦截它的绘制,并像 View
类那样做来淡出边缘来实现,但仅限于最终文本行的最后一段。
在这个例子中,我们创建了一个从透明到纯黑色的单位水平线性渐变。当我们准备绘制时,这个单位渐变被缩放到一个长度,计算为 TextView
的最终行长度的简单分数,然后相应地定位。
创建了一个离屏缓冲区,我们让 TextView
将其内容绘制到该缓冲区。然后,我们使用 PorterDuff.Mode.DST_OUT
的传输模式在其上绘制淡入淡出渐变,这实质上将底层内容清除到相对于给定点处渐变不透明度的一定程度。将该缓冲区绘制回屏幕上会导致所需的淡入淡出,无论背景是什么。
public class FadingTextView extends AppCompatTextView {
private static final float FADE_LENGTH_FACTOR = .4f;
private final RectF drawRect = new RectF();
private final Rect realRect = new Rect();
private final Path selection = new Path();
private final Matrix matrix = new Matrix();
private final Paint paint = new Paint();
private final Shader shader =
new LinearGradient(0f, 0f, 1f, 0f, 0x00000000, 0xFF000000, Shader.TileMode.CLAMP);
public FadingTextView(Context context) {
this(context, null);
}
public FadingTextView(Context context, AttributeSet attrs) {
this(context, attrs, android.R.attr.textViewStyle);
}
public FadingTextView(Context context, AttributeSet attrs, int defStyleAttribute) {
super(context, attrs, defStyleAttribute);
paint.setShader(shader);
paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT));
}
@Override
protected void onDraw(Canvas canvas) {
// Locals
final RectF drawBounds = drawRect;
final Rect realBounds = realRect;
final Path selectionPath = selection;
final Layout layout = getLayout();
// Figure last line index, and text offsets there
final int lastLineIndex = getLineCount() - 1;
final int lastLineStart = layout.getLineStart(lastLineIndex);
final int lastLineEnd = layout.getLineEnd(lastLineIndex);
// Let the Layout figure a Path that'd cover the last line text
layout.getSelectionPath(lastLineStart, lastLineEnd, selectionPath);
// Convert that Path to a RectF, which we can more easily modify
selectionPath.computeBounds(drawBounds, false);
// Naive text direction determination; may need refinement
boolean isRtl =
layout.getParagraphDirection(lastLineIndex) == Layout.DIR_RIGHT_TO_LEFT;
// Narrow the bounds to just the fade length
if (isRtl) {
drawBounds.right = drawBounds.left + drawBounds.width() * FADE_LENGTH_FACTOR;
} else {
drawBounds.left = drawBounds.right - drawBounds.width() * FADE_LENGTH_FACTOR;
}
// Adjust for drawables and paddings
drawBounds.offset(getTotalPaddingLeft(), getTotalPaddingTop());
// Convert drawing bounds to real bounds to determine
// if we need to do the fade, or a regular draw
drawBounds.round(realBounds);
realBounds.offset(-getScrollX(), -getScrollY());
boolean needToFade = realBounds.intersects(getTotalPaddingLeft(), getTotalPaddingTop(),
getWidth() - getTotalPaddingRight(), getHeight() - getTotalPaddingBottom());
if (needToFade) {
// Adjust and set the Shader Matrix
final Matrix shaderMatrix = matrix;
shaderMatrix.reset();
shaderMatrix.setScale(drawBounds.width(), 1f);
if (isRtl) {
shaderMatrix.postRotate(180f, drawBounds.width() / 2f, 0f);
}
shaderMatrix.postTranslate(drawBounds.left, drawBounds.top);
shader.setLocalMatrix(shaderMatrix);
// Save, and start drawing to an off-screen buffer
final int saveCount;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
saveCount = canvas.saveLayer(null, null);
} else {
saveCount = canvas.saveLayer(null, null, Canvas.ALL_SAVE_FLAG);
}
// Let TextView draw itself to the buffer
super.onDraw(canvas);
// Draw the fade to the buffer, over the TextView content
canvas.drawRect(drawBounds, paint);
// Restore, and draw the buffer back to the Canvas
canvas.restoreToCount(saveCount);
} else {
// Regular draw
super.onDraw(canvas);
}
}
}
这是 TextView
的直接替代品,您可以类似地在布局中使用它。
<com.example.app.FadingTextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="#e2f3eb"
android:textColor="#0b8043"
android:lineSpacingMultiplier="1.2"
android:text="@string/umang" />
注意事项:
淡入淡出长度的计算基于最后一行文本长度的常数分数,此处由 FADE_LENGTH_FACTOR
确定。这似乎与 Play Store 组件的基本方法相同,因为淡入淡出的绝对长度似乎随行长度而变化。 FADE_LENGTH_FACTOR
值可以根据需要进行更改。
FadingTextView
目前扩展了 AppCompatTextView
,但如果您需要的话,它可以作为普通的 TextView
完美运行。我认为它也可以用作 MaterialTextView
,尽管我还没有对其进行彻底的测试。
这个例子主要针对比较普通的使用;即,作为一个简单的包装静态标签。虽然我已经尝试考虑并测试我能想到的每一个 TextView
设置,这可能会影响这一点——例如,复合可绘制对象、填充、可选文本、滚动、文本方向和对齐等——我不能保证我已经想到了一切。
关于android - 如何淡出 TextView 中最后一行的结尾?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49295526/
我在activity_main.xml中有一个Textview设计,我想用它来在另一个textview下面显示一个textview。我从 stackoverlow 本身获取了这段代码,并做了一些更改以
伙计们,我尝试添加另一个 TextView ,但是当我尝试时,新的 TextView 已放置在前一个 TextView 上,如何使第二个 TextView 作为新的第二个 TextView 下降 这是
现在 textview 显示如果我将滚动到底部的语句。看起来 promis textview 变大了。有办法解决吗? 星号下的所有内容都是 promis textview 的一部分。它是一个带有多个换
我有多个 TextView 。所有 TextView 都是一个单词。我想用它们来构建一个段落。但他们不应该失去他们的 TextView 功能。因此它们不应该用作字符串。因为那样的话,我希望它们可以一一
附件是指向我的应用程序屏幕截图的链接和显示我想要实现的功能的视频的链接。基本上我想要的是当我单击任何功能区上的“更多”按钮时,它应该自行展开并且用从 JSON 文件中获取的数据替换文本。功能区的高度应
我想显示 edittext 中的文本,但由于输入了大量字符,布局被破坏了。我如何限制在 textview 上显示的字符数 最佳答案 你可以尝试在 Edittext actionlistener 上使用
是否可以将 textview 环绕在 textview 周围,第二个 textview 将环绕到第一个 textview 下的下一行? 例如: 我尝试过使用 android:layout_weig
假设我有 10 个文本项,所有这些项都对用户可见。每个文本项都有不同的颜色和样式。我不知道实现此目标的最佳方法是什么。 多个静态 TextView - 最容易实现,但性能可能最差。 TextView
我正在尝试创建一个布局,其中有两个 TextViews 彼此相邻。第一个 TextView 具有可变长度并且是单行的。第二个 TextView 包含一个固定标签,应该不被省略。第二个标签必须紧挨着第一
我正在使用 onTouchEvent 方法来缩放 TextView ,但是当我有两个 TextView 并且我只想调整第一个 TextView 而不是第二个 TextView 时,我遇到了麻烦。我的意
在我的 android 应用程序中,我有一个自动完成 TextView 字段,可以向用户建议不同的州名称。代码如下: ArrayAdapter state_adapter=new ArrayAdap
我怎样才能让 textview 开始一个新行,一直从左边开始,同时保留属性“toTheRightOf”username25? 像这样: 最佳答案 我的猜测是 twitch 应
我正在做一个动态 TextView 的例子,它是通过按下一个按钮创建的。它们的创建格式类似于: 1- Textview (fist created) 2- TextView 3- TextView (
我想要 TextView 和描述的自动调整大小的文本 here .我想,“测试”这个词应该在单行上对齐,但它是基于字符包装的(参见另一行的“g”)。我应该如何更改我的配置,以便一切都按我的意愿工作?
我有一个简单的布局,在两个 ImageView 之间包含一个 TextView。 当我在运行时使用“setText()”更改 TextView 时,它发生了变化,但失去了“center_horizo
作为 Android 初学者,我正在尝试显示一棵树。我已经找到了如何以编程方式添加新的 TextView,但我不知道如何放置它们,我的意思是如何根据父节点设置它们的填充/边距。 非常感谢您的回答, 巴
对于 AndroidTV,我创建了一个简单的搜索栏和 TextView 。我正在尝试对齐搜索栏拇指的 TextView 并尝试随着进度移动。 尝试了这些解决方案,但不起作用 link1 , link2
我是 Android Studio 新手,正在编写应用程序。 我正在尝试它,我想知道如何在 Textview1 包含某些内容的情况下启用(例如)Textview2。 我尝试了这段代码,但它不起作用:
我从这里获取 TextView 实例: TextView date = null; try { date = (TextView) getLayoutI
这是在别处被问到的,但解决方案对我不起作用。所以用更多的背景再次摆姿势。问题是一个 Activity 包含一个滚动的音乐标题 TextView ,它被更新的耗时计数器 TextView 中断。 我的
我是一名优秀的程序员,十分优秀!