2016-08-21 13 views
4

私は大きなテキストをページし、各単語や文に複数のスパンを設定するアプリを持っています。私は各単語の背景を描くためにReplacementSpanを使用しています。私はBackgroundSpanを使用することができません。単純すぎてキャンバスをコントロールすることができないからです。 ReplacementSpanがMetricAffectingSpanを拡張しているので、テキストのレイアウトに影響を与え、ページングを完全に破棄します。私は各ページのテキストを計算するためにStaticLayoutを使用しています。また、StaticLayoutはスパニングの影響を先験的に計算できるようにスパニングを許可していません。AndroidのReplacementSpanに代わるもの

ReplacementSpanの代替品はありますか?テキスト自体のサイズやレイアウトに影響を与えずに、どのように背景を描くことができますか?

これは私のreplacementspanのコードです:

public class BackgroundColorWithoutLineHeightSpan extends ReplacementSpan { 

    private static final float DP_ACTIVE = ViewsUtils.dpToPx(4); 
    private static final int DP_OUTSIDE_PADDING = (int) ViewsUtils.dpToPx(6); 
    private static final float DP_PHRASE = ViewsUtils.dpToPx(4); 
    private static final float DP_ROUNDED = ViewsUtils.dpToPx(3); 

    private final int mColor; 
    private final int mTextHeight; 
    private int mBorderColor; 
    private boolean mIsSelected; 
    private boolean mIsPhrase; 

    public BackgroundColorWithoutLineHeightSpan(int color, int textHeight, boolean isPhrase) { 
    mColor = color; 
    mTextHeight = textHeight; 
    mIsPhrase = isPhrase; 
    } 

    public BackgroundColorWithoutLineHeightSpan(int color, int textHeight, boolean isSelected, int borderColor, boolean isPhrase) { 
    mColor = color; 
    mTextHeight = textHeight; 
    mIsSelected = isSelected; 
    mBorderColor = borderColor; 
    mIsPhrase = isPhrase; 
    } 

    @Override 
    public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, Paint.FontMetricsInt fm) { 
    return Math.round(measureText(paint, text, start, end)); 
    } 

    @Override 
    public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, Paint paint) { 

    canvas.save(); 

    Rect newRect = canvas.getClipBounds(); 
    newRect.inset(-DP_OUTSIDE_PADDING, -DP_OUTSIDE_PADDING); 

    canvas.clipRect(newRect, Region.Op.REPLACE); 

    float measuredText = measureText(paint, text, start, end); 

    int paintColor = paint.getColor(); 

    if (!mIsSelected) { 
     RectF rect; 
     rect = new RectF(x, top, x + measuredText, top + mTextHeight); 

     paint.setStrokeWidth(0.0f); 
     paint.setColor(mColor); 
     paint.setStyle(Paint.Style.FILL); 

     canvas.drawRoundRect(rect, DP_ROUNDED, DP_ROUNDED, paint); 

    } else { 

     RectF rect; 
     if (mIsPhrase) { 
     rect = new RectF(x - DP_PHRASE, top - DP_PHRASE, x + measuredText + DP_PHRASE, top + mTextHeight + DP_PHRASE); 
     } else { 
     rect = new RectF(x - DP_ACTIVE, top - DP_ACTIVE, x + measuredText + DP_ACTIVE, top + mTextHeight + DP_ACTIVE); 
     } 
     paint.setStrokeWidth(0.0f); 
     paint.setColor(mColor); 
     paint.setStyle(Paint.Style.FILL); 

     canvas.drawRoundRect(rect, DP_ROUNDED, DP_ROUNDED, paint); 

     RectF border; 
     if (mIsPhrase) { 
     border = new RectF(x - DP_PHRASE, top - DP_PHRASE, x + measuredText + DP_PHRASE, top + mTextHeight + DP_PHRASE); 
     } else { 
     border = new RectF(x - DP_ACTIVE, top - DP_ACTIVE, x + measuredText + DP_ACTIVE, top + mTextHeight + DP_ACTIVE); 
     } 

     paint.setColor(mBorderColor); 
     paint.setStrokeWidth(4.0f); 
     paint.setStyle(Paint.Style.STROKE); 

     canvas.drawRoundRect(border, DP_ROUNDED, DP_ROUNDED, paint); 
    } 

    paint.setStyle(Paint.Style.FILL); 
    paint.setColor(paintColor); 
    canvas.drawText(text, start, end, x, y, paint); 

    canvas.restore(); 
    } 

    private float measureText(Paint paint, CharSequence text, int start, int end) { 
    return paint.measureText(text, start, end); 
    } 
} 

答えて

1

は、この単純なスパンをしようと、それはすべてのスパンで赤一色の背景を描画します(これは複数行のスパンであっても)しかし、あなたは、あなたが好きな描くことができます:

class LBS implements LineBackgroundSpan { 
    private final TextView tv; 
    private int start; 
    private int end; 

    public LBS(TextView tv, int start, int end) { 
     this.tv = tv; 
     this.start = start; 
     this.end = end; 
    } 

    @Override 
    public void drawBackground(Canvas c, Paint p, int left, int right, int top, int baseline, int bottom, CharSequence text, int start, int end, int lnum) { 
     Layout layout = tv.getLayout(); 
     int startLine = layout.getLineForOffset(this.start); 
     int endLine = layout.getLineForOffset(this.end); 
     if (startLine <= lnum && lnum <= endLine) { 
      if (startLine == lnum) { 
       left = (int) layout.getPrimaryHorizontal(this.start); 
      } 
      if (endLine == lnum) { 
       right = (int) layout.getPrimaryHorizontal(this.end); 
      } 
      int origColor = p.getColor(); 
      p.setColor(Color.RED); 
      c.drawRect(left, top, right, bottom, p); 
      p.setColor(origColor); 
     } 
    } 
} 

テストコード(startendとして0ssb.length()を設定することは非常に効率的ではありませんので、あなたがそれを最適化することができます):

TextView tv = new TextView(this); 
setContentView(tv); 
tv.setTextSize(32); 
SpannableStringBuilder ssb = new SpannableStringBuilder("Chop a handfull spinach, pork shoulder, and dill in a large cooker over medium heat, cook for six minutes and varnish with some bok choy."); 
LBS span = new LBS(tv, 30, 100); 
ssb.setSpan(span, 0, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); 
tv.setText(ssb); 

Log.d(TAG, "onCreate text [" + ssb.subSequence(30, 100) + "]"); 

EDIT

あなたはそれのように修正されたバージョンを使用することができますハイライト/マークする複数の単語がある場合:

class LBS implements LineBackgroundSpan { 
    TextView tv; 
    List<Pair<Integer, Integer>> ranges; 

    public LBS(TextView tv) { 
     this.tv = tv; 
     ranges = new ArrayList<>(); 
    } 

    public void add(int start, int end) { 
     ranges.add(new Pair<>(start, end)); 
    } 

    @Override 
    public void drawBackground(Canvas c, Paint p, int left, int right, int top, int baseline, int bottom, CharSequence text, int start, int end, int lnum) { 
     Layout layout = tv.getLayout(); 
     for (Pair<Integer, Integer> range : ranges) { 
      int startLine = layout.getLineForOffset(range.first); 
      int endLine = layout.getLineForOffset(range.second); 
      if (startLine <= lnum && lnum <= endLine) { 
       if (startLine == lnum) { 
        left = (int) layout.getPrimaryHorizontal(range.first); 
       } 
       if (endLine == lnum) { 
        right = (int) layout.getPrimaryHorizontal(range.second); 
       } 
       int origColor = p.getColor(); 
       p.setColor(Color.RED); 
       c.drawRect(left, top, right, bottom, p); 
       p.setColor(origColor); 
      } 
     } 
    } 
} 

テストコード:

TextView tv = new TextView(this); 
    setContentView(tv); 
    tv.setTextSize(32); 
    String text = "Chop a handfull spinach, pork shoulder, and dill in a large cooker over medium heat, cook for six minutes and varnish with some bok choy."; 
    SpannableStringBuilder ssb = new SpannableStringBuilder(text); 
    LBS span = new LBS(tv); 

    String[] words = { 
      "spinach, pork shoulder", "cooker", "with some bok choy", 
    }; 
    for (String word : words) { 
     int idx = text.indexOf(word); 
     span.add(idx, idx + word.length()); 
    } 

    ssb.setSpan(span, 0, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); 
    tv.setText(ssb); 
+0

はとてもあなたに感謝多くのあなたの助けに!このソリューションでは、各単語の背景を設定することができ、レイアウトに影響を与えることなくテキストが引き続き折り返されます。 – mobilepotato7

+0

ほんの数分前に見つかったことがもう1つあります: 'Layout#getSelectionPath'、それをチェックしてください。多分役に立つかもしれません。 – pskink

+0

レイアウトはとても強力です。私はテキストとは何か他の機能を持っている、私はレイアウトで他に何ができるかを確かにチェックします。ありがとう。 – mobilepotato7

関連する問題