解决TextView在显示文字的时候,一行还没显示满就跳到下一行(最新)

解决Android TextView排版问题
本文介绍了一种解决Android TextView中英文混排时出现的排版异常问题的方法。通过自定义TextView,确保了中文标点不会导致额外的空间留白,并且保持了良好的英文单词完整性。

转载请注明:转自 http://blog.youkuaiyun.com/u011176685/article/details/48295185

一、问题描述:

Android的TextView在显示文字的时候,如果有段中文有英文,有中文,有中文标点符号,你会发现,当要换行的时候遇到中文标点,

这一行就会空出很多空格出来。原因是:

1) TextView在显示中文的时候 标点符号不能显示在一行的行首和行尾,如果一个标点符号刚好在一行的行尾,该标点符号就会连同前一个字符跳到下一行显示;

2)一个英文单词不能被显示在两行中( TextView在显示英文时,标点符号是可以放在行尾的,但英文单词也不能分开 );

3)全角和半角的问题,汉字无论全角还是半角都是占2个字节,英文和符号在半角是占一个字节,全角是占两个字节。

二、解决方法

参考资料中:http://niufc.iteye.com/blog/1729792 

可能由于时间问题,都没有很好解决我的问题。将textview中的字符全角化没有效果,去除特殊字符或将所有中文标号替换为英文标号。这个有点效果,但是产品经理说文案不符合标准。改源代码担心出问题,影响其他的应用。自定义TextView时,canvas.setViewport()这个方法的api被删了。然后各种百度查资料,很多都是转过来转过去。然并卵。后面找了好久才找到一个靠谱的。完美的解决了我的问题。


自定义TextView(直接把代码拷进去就能用)

package com.lhx.widget;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.text.Layout;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.widget.TextView;

/**
 * @author lhx
 * @Date 9/8/15
 */
public class MyTextView extends TextView {

    private int mLineY;
    private int mViewWidth;

    public MyTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right,
                            int bottom) {
        super.onLayout(changed, left, top, right, bottom);
    }

    @Override
    protected void onDraw(Canvas Canvas) {
        TextPaint paint = getPaint();
        paint.setColor(getCurrentTextColor());
        paint.drawableState = getDrawableState();
        mViewWidth = getMeasuredWidth();
        String text = getText().toString();
        mLineY = 0;
        mLineY += getTextSize();
        Layout layout = getLayout();

        // layout.getLayout()在4.4.3出现NullPointerException
        if (layout == null) {
            return;
        }

        Paint.FontMetrics fm = paint.getFontMetrics();

        int textHeight = (int) (Math.ceil(fm.descent - fm.ascent));
        textHeight = (int) (textHeight * layout.getSpacingMultiplier() + layout
                .getSpacingAdd());
        //解决了最后一行文字间距过大的问题
        for (int i = 0; i < layout.getLineCount(); i++) {
            int lineStart = layout.getLineStart(i);
            int lineEnd = layout.getLineEnd(i);
            float width = StaticLayout.getDesiredWidth(text, lineStart,
                    lineEnd, getPaint());
            String line = text.substring(lineStart, lineEnd);

            if(i < layout.getLineCount() - 1) {
                if (needScale(line)) {
                    drawScaledText(Canvas, lineStart, line, width);
                } else {
                    Canvas.drawText(line, 0, mLineY, paint);
                }
            } else {
                Canvas.drawText(line, 0, mLineY, paint);
            }
            mLineY += textHeight;
        }
    }

    private void drawScaledText(Canvas Canvas, int lineStart, String line,
                                float lineWidth) {
        float x = 0;
        if (isFirstLineOfParagraph(lineStart, line)) {
            String blanks = "  ";
            Canvas.drawText(blanks, x, mLineY, getPaint());
            float bw = StaticLayout.getDesiredWidth(blanks, getPaint());
            x += bw;

            line = line.substring(3);
        }

        int gapCount = line.length() - 1;
        int i = 0;
        if (line.length() > 2 && line.charAt(0) == 12288
                && line.charAt(1) == 12288) {
            String substring = line.substring(0, 2);
            float cw = StaticLayout.getDesiredWidth(substring, getPaint());
            Canvas.drawText(substring, x, mLineY, getPaint());
            x += cw;
            i += 2;
        }

        float d = (mViewWidth - lineWidth) / gapCount;
        for (; i < line.length(); i++) {
            String c = String.valueOf(line.charAt(i));
            float cw = StaticLayout.getDesiredWidth(c, getPaint());
            Canvas.drawText(c, x, mLineY, getPaint());
            x += cw + d;
        }
    }

    private boolean isFirstLineOfParagraph(int lineStart, String line) {
        return line.length() > 3 && line.charAt(0) == ' '
                && line.charAt(1) == ' ';
    }

    private boolean needScale(String line) {
        if (line == null || line.length() == 0) {
            return false;
        } else {
            return line.charAt(line.length() - 1) != '\n';
        }
    }

}

xml文件代码

 <!--工作时间提醒-->
    <LinearLayout
            android:layout_width="match_parent"
            android:id="@+id/ll_chat"
            android:background="@color/topbar_chat"
            android:layout_height="wrap_content">
        <com.lhx.widget.MyTextView
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:id="@+id/tv_reminder"
                android:layout_marginTop="5dp"
                android:layout_marginBottom="5dp"
                android:layout_marginRight="8dp"
                android:layout_marginLeft="8dp"
                android:text = "服务商不管以任何形式要求线下交易,都存在诈骗的风险,请提高警惕。This is test!欢迎相互关注。有不对的地方望指出和包容。谢谢! "
                android:textColor="@color/text_orange_1"/>
    </LinearLayout>

以上就很好的解决了TextView中英文字符排版的问题,折腾了好久终于搞定。希望对您有点帮助。




<think>我们正在实现一个RecyclerView列表,每行显示文本和右侧箭头,点击跳转到另一个Activity。 步骤: 1. 准备数据源(字符串列表) 2. 创建RecyclerView的Item布局(包含一个TextView和一个ImageView作为箭头) 3. 创建Adapter和ViewHolder 4. 在Activity中设置RecyclerView(包括LayoutManager和Adapter) 5. 实现点击事件:在Adapter中为每个Item设置点击监听,点击时跳转到目标Activity 注意:这里我们使用RecyclerView来实现,因为RecyclerView是ListView的升级版,具有更好的性能和灵活性[^1]。 具体实现如下:</think>### Android Java:RecyclerView列表显示文本和右侧箭头(点击跳转Activity) #### 一、实现步骤概览 1. **创建列表项布局**(含文本和箭头) 2. **建立数据模型** 3. **实现RecyclerView适配器** 4. **设置点击跳转逻辑** 5. **在Activity中配置RecyclerView** #### 二、完整实现代码 ##### 1. 列表项布局 (`item_list.xml`) ```xml <!-- res/layout/item_list.xml --> <androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_margin="8dp" app:cardElevation="4dp"> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal" android:padding="16dp" android:clickable="true" android:focusable="true"> <TextView android:id="@+id/tvItemText" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:textSize="18sp" android:textColor="@color/black"/> <ImageView android:id="@+id/ivArrow" android:layout_width="24dp" android:layout_height="24dp" android:src="@drawable/ic_arrow_right" <!-- 准备向右箭头图标 --> android:contentDescription="跳转箭头"/> </LinearLayout> </androidx.cardview.widget.CardView> ``` ##### 2. 数据模型类 ```java public class ListItem { private String title; private Class<?> targetActivity; // 目标Activity类 public ListItem(String title, Class<?> targetActivity) { this.title = title; this.targetActivity = targetActivity; } // Getters public String getTitle() { return title; } public Class<?> getTargetActivity() { return targetActivity; } } ``` ##### 3. 适配器实现 (Adapter) ```java public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> { private final List<ListItem> dataList; private final Context context; public MyAdapter(Context context, List<ListItem> dataList) { this.context = context; this.dataList = dataList; } public static class ViewHolder extends RecyclerView.ViewHolder { TextView tvText; ImageView ivArrow; public ViewHolder(View itemView) { super(itemView); tvText = itemView.findViewById(R.id.tvItemText); ivArrow = itemView.findViewById(R.id.ivArrow); } } @Override public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { View view = LayoutInflater.from(parent.getContext()) .inflate(R.layout.item_list, parent, false); return new ViewHolder(view); } @Override public void onBindViewHolder(ViewHolder holder, int position) { ListItem item = dataList.get(position); holder.tvText.setText(item.getTitle()); // 设置点击事件 holder.itemView.setOnClickListener(v -> { Intent intent = new Intent(context, item.getTargetActivity()); context.startActivity(intent); // 添加跳转动画 if (context instanceof Activity) { ((Activity) context).overridePendingTransition( R.anim.slide_in_right, R.anim.slide_out_left); } }); } @Override public int getItemCount() { return dataList.size(); } } ``` ##### 4. Activity中配置RecyclerView ```java public class MainActivity extends AppCompatActivity { private RecyclerView recyclerView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 初始化Toolbar(可选) Toolbar toolbar = findViewById(R.id.toolbar); setSupportActionBar(toolbar); // 配置RecyclerView recyclerView = findViewById(R.id.rvList); recyclerView.setLayoutManager(new LinearLayoutManager(this)); // 准备数据 List<ListItem> items = new ArrayList<>(); items.add(new ListItem("设置", SettingsActivity.class)); items.add(new ListItem("个人资料", ProfileActivity.class)); items.add(new ListItem("消息中心", MessageActivity.class)); // 设置适配器 MyAdapter adapter = new MyAdapter(this, items); recyclerView.setAdapter(adapter); // 添加分割线 recyclerView.addItemDecoration(new DividerItemDecoration( this, DividerItemDecoration.VERTICAL)); } } ``` #### 三、关键优化点 1. **跳转动画效果**(创建动画资源): ```xml <!-- res/anim/slide_in_right.xml --> <set xmlns:android="http://schemas.android.com/apk/res/android"> <translate android:fromXDelta="100%" android:toXDelta="0" android:duration="300"/> </set> <!-- res/anim/slide_out_left.xml --> <set xmlns:android="http://schemas.android.com/apk/res/android"> <translate android:fromXDelta="0" android:toXDelta="-100%" android:duration="300"/> </set> ``` 2. **防止快速重复点击**: ```java // 在适配器中添加防抖机制 private long lastClickTime = 0; private static final long CLICK_DELAY = 500; // 500毫秒间隔 // 修改点击事件处理 holder.itemView.setOnClickListener(v -> { long currentTime = SystemClock.elapsedRealtime(); if (currentTime - lastClickTime < CLICK_DELAY) return; lastClickTime = currentTime; // ...跳转逻辑 }); ``` 3. **导航栏整合**(参考引用[3]): ```java // 在onCreate中配置导航抽屉 mDrawerLayout = findViewById(R.id.drawer); ActionBar actionBar = getSupportActionBar(); if (actionBar != null) { actionBar.setDisplayHomeAsUpEnabled(true); actionBar.setHomeAsUpIndicator(R.drawable.ic_drawer); } ``` #### 四、效果说明 实现功能 | 实现方式 | 用户交互效果 ---|---|--- 文本显示 | `TextView` | 左侧显示标题文本 箭头图标 | `ImageView` | 右侧显示跳转指示 点击跳转 | `setOnClickListener` | 点击整行触发跳转 跳转动效 | `overridePendingTransition` | 平滑的右滑进入效果 导航整合 | `Toolbar + DrawerLayout` | 支持侧滑菜单[^3] > **注意**:RecyclerView通过RecycleBin机制高效管理视图内存,即使处理大量数据也不会OOM[^1]。使用`CardView`可增强列表项的立体感和点击反馈。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值