草庐IT

java - Android实现了一个带有两个clip textview的progressBar

coder 2023-12-22 原文

我要实现一个带有两个剪辑 TextView 的进度条。 就像是:

我使用了一些技巧(使用paddingEnd/paddingStart,back/front textview来实现textview的剪辑效果)来实现它:

    // part of activity
    TextView leftFrontText;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        View view = View.inflate(this, R.layout.test_page_layout, containerLayout);

        View progressBar = view.findViewById(R.id.progress_bar);
        leftFrontText = view.findViewById(R.id.left_front_textview);

        int progress = 59;
        ValueAnimator va = ValueAnimator.ofFloat(0f, progress);
        int mDuration = 1000;
        va.setDuration(mDuration);
        va.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            public void onAnimationUpdate(ValueAnimator animation) {
                float value = (float)animation.getAnimatedValue();
                setViewWidth(progressBar, (int)value);
                setLeftFrontTextViewPadding((int)value);
            }
        });
//        va.addListener(new Animator.AnimatorListener() {
//            @Override
//            public void onAnimationStart(Animator animation) {
//
//            }
//
//            @Override
//            public void onAnimationEnd(Animator animation) {
//                setLeftFrontTextViewPadding(progress);
//            }
//
//            @Override
//            public void onAnimationCancel(Animator animation) {
//
//            }
//
//            @Override
//            public void onAnimationRepeat(Animator animation) {
//
//            }
//        });
        va.start();
    }

    private void setViewWidth(View view, int dp) {
        ViewGroup.LayoutParams layoutParams = view.getLayoutParams();
        layoutParams.width = ThemeUtil.dpToPx(this, dp);
        view.setLayoutParams(layoutParams);
    }

    private void setLeftFrontTextViewPadding(int progressBarDp) {

        int marginStart = ThemeUtil.dpToPx(TestPageActivity.this, 12);
        leftFrontText.post(new Runnable() {
            @Override
            public void run() {
                int textViewLength = leftFrontText.getWidth();
                int progressbarDistance = ThemeUtil.dpToPx(TestPageActivity.this, progressBarDp);
                if (textViewLength >= 0) {
                    int padding = marginStart + textViewLength - progressbarDistance;
                    leftFrontText.setPadding(0, 0, padding > 0 ? -padding : 0, 0);
                }
            }
        });

    }

布局是:

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="28dp"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:background="@drawable/rounded_corner_bg_for_cashlfow_homepage_category_cell"
    android:layout_marginStart="16dp"
    android:layout_marginEnd="16dp"
    >

    <TextView
        android:id="@+id/left_bottom_textview"
        android:layout_width="wrap_content"
        android:layout_height="0dp"
        android:gravity="center_vertical"
        android:text="Entertatinment"
        android:textSize="@dimen/main_text_size"
        android:maxLines="1"
        android:ellipsize="end"
        android:textColor="@color/grey1"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        android:layout_marginStart="12dp"
        />

    <View
        android:id="@+id/progress_bar"
        android:layout_width="10dp"
        android:layout_height="match_parent"
        android:background="@drawable/rounded_corner_front_progreebar_homepage_category_cell"
        app:layout_constraintStart_toStartOf="parent"
        />

    <TextView
        android:id="@+id/left_front_textview"
        android:layout_width="wrap_content"
        android:layout_height="0dp"
        android:gravity="center_vertical|start"
        android:text="Entertatinment"
        android:textSize="@dimen/main_text_size"
        android:maxLines="1"
        android:ellipsize="end"
        android:textColor="@color/white"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        android:layout_marginStart="12dp"
        />

    <TextView
        android:id="@+id/right_bottom_amount_textview"
        android:layout_width="wrap_content"
        android:layout_height="0dp"
        android:gravity="center_vertical|end"
        android:text="$0 / $0"
        android:textSize="@dimen/main_text_size"
        android:maxLines="1"
        android:textColor="@color/white"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        android:layout_marginEnd="12dp"
        />

    <TextView
        android:id="@+id/right_front_amount_textview"
        android:layout_width="wrap_content"
        android:layout_height="0dp"
        android:gravity="center_vertical|end"
        android:text="$0 / $0"
        android:textSize="@dimen/main_text_size"
        android:maxLines="1"
        android:textColor="@color/black1"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        android:layout_marginEnd="12dp"
        android:paddingStart="0dp"
        />

</android.support.constraint.ConstraintLayout>

当我运行这段代码时,它几乎达到了我想要的效果。但它有一个问题。 Textview在做动画的时候一直在闪烁,当动画结束的时候,textview的颜色有时会变成全白。

我认为 setLeftFrontTextViewPaddingleftFrontText.post() 可能有问题,比如我如何获取 textview 的宽度。但我不太确定。谁能告诉我正确的方法吗?或者任何其他更好的方法来实现这种进度条。

最佳答案

为什么不尝试这种方法并利用 ConstraintLayout 的强大功能作为您的进度条:

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/parentview"
    android:layout_width="360dp"
    android:layout_height="28dp"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:background="#ddddff"
    android:layout_marginStart="16dp"
    android:layout_marginEnd="16dp">

    <TextView
        android:id="@+id/left_bottom_textview"
        android:layout_width="180dp"
        android:layout_height="match_parent"
        android:gravity="center_vertical"
        android:text="Entertainment"
        android:maxLines="1"
        android:ellipsize="end"
        android:textColor="#404040"
        app:layout_constraintStart_toStartOf="parent"/>

    <TextView
        android:id="@+id/right_front_amount_textview"
        android:layout_width="180dp"
        android:layout_height="0dp"
        android:gravity="center_vertical|end"
        android:text="$0 / $0"
        android:maxLines="1"
        android:textColor="@color/black"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent" />

    <android.support.constraint.ConstraintLayout
        android:id="@+id/progress_bar"
        android:layout_width="50dp"
        android:layout_height="match_parent"
        app:layout_constraintStart_toStartOf="parent" 
        android:background="#004090">
        <TextView
            android:id="@+id/left_front_textview"
            android:layout_width="180dp"
            android:layout_height="wrap_content"
            android:text="Entertainment"
            android:singleLine="true"
            android:textColor="#fff"
            android:ellipsize="none"
            android:layout_paddingStart="12dp"
            android:layout_paddingLeft="12dp"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintStart_toStartOf="parent"/>  
        <TextView
            android:id="@+id/right_bottom_amount_textview"
            android:layout_width="180dp"
            android:layout_height="wrap_content"
            android:layout_centerVertical="true"
            android:gravity="center_vertical|end"
            android:text="$0 / $0"
            android:layout_alignRight="@id/left_front_textview"
            android:textColor="#ffffff"
            android:layout_paddingEnd="12dp"
            android:layout_paddingRight="12dp"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintStart_toEndOf="@id/left_front_textview"/>
    </android.support.constraint.ConstraintLayout>  

</android.support.constraint.ConstraintLayout>

所以基本上我们会:

父布局(父 View ):具有固定的 layout_width 360dp 并包含初始的 Grey TextViews

Progress 布局 (progress_bar): 也具有固定大小 360dp 并包含前面的 White TextViews

因此我们将 Progress 布局 (progress_bar) 叠加到其父布局之上,并增加或减少 layout_width 以模拟不闪烁的进度:

下图显示了静态 layout_width 为 74dp 的进度布局;另请注意,根据上面提供的示例,progress layout_width 的范围可以从 1dp 到 max:360dp父 View 宽度的 1% 到 100% .

关于java - Android实现了一个带有两个clip textview的progressBar,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57555634/

有关java - Android实现了一个带有两个clip textview的progressBar的更多相关文章

  1. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  2. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  3. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  4. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  5. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

  6. ruby - 为什么 SecureRandom.uuid 创建一个唯一的字符串? - 2

    关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?

  7. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  8. ruby - 如何根据特征实现 FactoryGirl 的条件行为 - 2

    我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden

  9. ruby-on-rails - Rails - 从另一个模型中创建一个模型的实例 - 2

    我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案

  10. ruby - 用 Ruby 编写一个简单的网络服务器 - 2

    我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b

随机推荐