Android TextView Justify Text


396

aのテキストをどのTextViewように正当化しますか(テキストを左側と右側にフラッシュします)?

ここで可能な解決策を見つけましたが、それは機能しません(vertical-centerをcenter_verticalなどに変更しても)。


@Jimbo答えはdefinetlyのために働いて正しいです私の場合のinputTextのTextView左入力と表示に右からアラビア語のためではなく、入力テキストのために、私はまた、重力=「右」を追加する必要がありました
SHAREEF

回答:


240

Androidが正当化を完全にサポートしているとは思いません。

UPDATE 2018-01-01:Android 8.0以降では、を使用して位置揃えモードをTextViewサポートしています


5
さらに分析すると、android:gravity = "fill_horizo​​ntal"を試してみることができます。これは「必要に応じてオブジェクトの水平方向のサイズを拡大して、コンテナを完全に埋める」と説明されていますが、テキストがどのように「拡大」されるかはわかりません。
CommonsWare 2009

8
android:gravity = "fill_horizo​​ntal"も機能しませんでした。結局のところ、Androidは正当化をサポートしていないようです。

6
いいえ、重力のようなプロパティを設定することはできません。ただし、テキストビューの代わりにWebビューを使用することで、テキストの正当性を設定できます。seal.io/2010/12/only-way-how-to-align-text-in-block-in.htmlを参照してください。(stackoverflow.com/questions/5976627/…から盗んだ)
jcaruso 2013年

2
@CommonsWareこれで、テキストを正当化する適切な方法はありますか?
John R

1
男、私はこれを達成するために重いwebviewで生活しています。そして、UIはまだAPIに追加されていないいくつかの新しいものを求めています。
nobalG 2016

156

@CommonsWareの答えは正しいです。Android 8.0以降は、「完全な正当化」(または、あいまいに呼ばれることがあるので、単に「正当化」)をサポートしています。

Androidは、「左揃え/右揃え」もサポートしています。区別については、正当化に関するウィキペディアの記事を参照してください。多くの人々は、「ジャスティフィケーション」の概念を完全なジャスティフィケーションだけでなく左/右のテキストの配置も含むと考えています。これは、左/右のテキストの配置をしたいときに検索することになります。この回答では、テキストの左/右揃えを実現する方法について説明します。

左揃え/右揃えのテキストを揃えることができます(質問が尋ねているように、完全な位置揃えとは対照的です)。例として、基本的な2列のフォーム(左側の列のラベルと右側の列のテキストフィールド)を例として使用します。この例では、左の列のラベルのテキストが右揃えになり、右の列のテキストフィールドと同じ高さに表示されます。

XMLレイアウトでは、すべてのTextView内に次の属性を追加することで、TextView要素自体(左の列)を右に揃えることができます。

<TextView
   ...
   android:layout_gravity="center_vertical|end">
   ...
</TextView>

ただし、テキストが複数行に折り返される場合でも、テキストはTextView内で左揃えに揃えられます。次の属性を追加すると、実際のテキストはTextView内で右揃え(左寄せ)になります。

<TextView
   ...
   android:gravity="end">
   ...
</TextView>

したがって、gravity属性は、TextView内のテキストを配置する方法を指定します。layout_gravityは、TextView要素自体を配置/レイアウトする方法を指定します。


12
私が正しく理解し、これをテストした結果を考えると、これはテキストを左または右に揃えることだけです。これはテキストを正当化しませんか?
Paul Lammertsma、2009

14
優秀な。追加するだけで、中央揃えが必要な場合は行うことができます android:layout_gravity="center_horizontal|center" android:gravity="center"
Luis A. Florit

入力テキストとアラビア語の右から左への入力と表示のinputtexttextviewに関する私のケースで明確に機能しています
shareef

1
これは単なる整合であり、正当化ではありません。ウィキペディアのリンクをよく読んでください。さまざまな種類の位置揃えの違いは、段落の最終行にのみ影響します。1行しかない段落には、左/右/中央揃えはありません。
Karu、2016年

それでは、なぜそうでない場合でもここで答えるのですjustify
user924

136

Androidでテキストを正当化するために私はWebViewを使用しました

    setContentView(R.layout.main);

    WebView view = new WebView(this);
    view.setVerticalScrollBarEnabled(false);

    ((LinearLayout)findViewById(R.id.inset_web_view)).addView(view);

    view.loadData(getString(R.string.hello), "text/html; charset=utf-8", "utf-8");

とhtml。

<string name="hello">
<![CDATA[
<html>
 <head></head>
 <body style="text-align:justify;color:gray;background-color:black;">
  Lorem ipsum dolor sit amet, consectetur 
  adipiscing elit. Nunc pellentesque, urna
  nec hendrerit pellentesque, risus massa
 </body>
</html>
]]>
</string>

それを証明する画像をアップロードすることはまだできませんが、「私にとってはうまくいきます」。


3
ここで素晴らしい解決策。FWIWあなたは余分なhtmlのほとんどを必要としません。テキストを揃えたbodyタグで十分です。
gnac

5
これはうまくいきます。あなたは以下のことで背景を透明にすることに注意してくださいview.loadData()view.setBackgroundColor("#00000000")
Paul Lammertsma、2011

ただし、カスタムフォント/書体をロードすることはできません。私はこれこの提案を試みましたが、運がありませんでした。
ポールランメルツマ

2
これらのスレッドで述べたように、解決策を見つけました。HTMLファイルを作成してアセットに配置すると、view.loadUrl()作品を介してロードされますが、そうではありview.loadData()ません。なぜ後者がそうしないのか私には手がかりがありません。
Paul Lammertsma、2011

1
@PaulLammertsma、setBackgroundColor(0x00000000)は、透明な背景を設定するための正しい形式です。
リッチー2013年

100

更新しました

このための簡単なクラスを作成しました。あなたが探しているものを達成するために現在2つの方法があります。どちらもWEBVIEWを必要とせサポートを提供します。

ライブラリhttps : //github.com/bluejamesbond/TextJustify-Android

SUPPORTS:5.XへのAndroid 2.0

セットアップ

// Please visit Github for latest setup instructions.

スクリーンショット

Comparison.png


本当に役に立ちますが、それを使用すると、私のTextViewsは元の形式を維持できません。マージン、テキストスタイル、およびテキストサイズもそうではないと思います。Pleseは引き続き機能しているので、非常に役立つはずです
Leonardo Sapuy

まあ私はそれらのクラスを確立することはできません。そのうちの1つにはパッケージ名がなく、もう1つには黄色のエラーが表示されます。実は信用できません。
メフメット2014

素晴らしいlibですが、このライブラリを使用してテキストに書式を追加する方法はまだわかりません。
セマンティカー2014年

4
このすばらしい共有ライブラリに感謝しますが、ペルシャ語またはアラビア語のテキストはサポートできません。方向を設定すると、単語が最初から最後ではなく、最後から最初に向かって描画されます。私がこれを意味します。私の言葉は: "سلام"であり、次のように描きます: "مالس"。(ペルシア語が理解できない場合は、この例を参照してください:「1234」->「4321」とさせてください)
うずまきナルト

1
scrollViewに基づく...しかし、素晴らしい解決策は、textviewでそれを可能にする答えをまだ見つけることができません。:(
superUser

88

TextViewAndroid O完全な正当化(新しい活版印刷の配置)自体を提供します。

あなたはこれを行う必要があります:

コトリン

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
    textView.justificationMode = JUSTIFICATION_MODE_INTER_WORD
}

ジャワ

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
    textView.setJustificationMode(JUSTIFICATION_MODE_INTER_WORD);
}

デフォルトはJUSTIFICATION_MODE_NONEです。


3
それがサポートライブラリに移植されて、それがOであることを期待しましょうO :)
Stefan Haustein

2
ここにライブラリを追加してください!!
Kunal Dharaiya

4
XMLを使用して正当化する方法?
Vikash Parajuli 2018年

14
xml:でandroid:justificationMode = "inter_word"を使用できます。
クリスチャンD

5
android:justificationModeにはAPI 26が必要です。
Bink

42

githubでJustifiedTextView for Androidプロジェクトを使用できます。これは、正当なテキストをシミュレートするカスタムビューです。Android 2.0以上と右から左への言語をサポートしています。 ここに画像の説明を入力してください


スパン可能な文字列はサポートされていません
MSepehr

どうすれば独自のテキストを追加できますか?
Karan 2015

githubのサンプルをご覧ください。
Saeed Zarinfam、2015

こんにちはサイード、あなたの助けのためのtnx、スパン可能なテキストビューをサポートする方法はありますか?!
Hamid Reza

uがこの質問のヘルプを私にしプラザう@SaeedZarinfam私は「Android向けJustifiedTextView」を使用しようとしたが、私は、XMLタグir.noghteh.JustifiedTextViewにエラーが発生しましたstackoverflow.com/questions/37911376/...
朱蒙(チュモン)

30

私はそれを行うためにネイティブのテキストビューに基づくウィジェットを書きます。

github


私はこれをお勧めします。これは、Android公式SDKの元のテキストビューに基づいているためです。私の個人的な意見では、この一般的なトピックに関して多くの人が投稿しているWebビューの手法よりもはるかに軽量です。たとえばメモリを賢くする必要があるアプリを作成する場合は、たとえばリストビューオブジェクトを使用して、次のようなものを使用することを検討してください。Ï試してみて、期待どおりに動作すること。あなたがこの1のような別のより良いものを知っている場合は、私とあなたの経験を共有してください。
スーパーユーザー

いい仕事だね 私が探していたもの。
スーパーユーザー

5
ペルシャ語のようなRTL言語をサポートしていません
穴の中で発砲します

1
@フランクチェン非常に便利なライブラリ。段落の終わりにたくさんのスペースがあります。どうすれば修正できますか?
iSrinivasan27 2016

1
私のために働いたが、textviewの最後の行が途切れました。私は、textviewのために5を詰め続けなければなりませんでした。
TharakaNirmana 2016年

23

私はこの問題を解決する方法を見つけましたが、これはあまり恵みではないかもしれませんが、効果は悪くありません。

その原理は、各行のスペースを固定幅のImageSpan(色は透明)に置き換えることです。

public static void justify(final TextView textView) {

    final AtomicBoolean isJustify = new AtomicBoolean(false);

    final String textString = textView.getText().toString();

    final TextPaint textPaint = textView.getPaint();

    final SpannableStringBuilder builder = new SpannableStringBuilder();

    textView.post(new Runnable() {
        @Override
        public void run() {

            if (!isJustify.get()) {

                final int lineCount = textView.getLineCount();
                final int textViewWidth = textView.getWidth();

                for (int i = 0; i < lineCount; i++) {

                    int lineStart = textView.getLayout().getLineStart(i);
                    int lineEnd = textView.getLayout().getLineEnd(i);

                    String lineString = textString.substring(lineStart, lineEnd);

                    if (i == lineCount - 1) {
                        builder.append(new SpannableString(lineString));
                        break;
                    }

                    String trimSpaceText = lineString.trim();
                    String removeSpaceText = lineString.replaceAll(" ", "");

                    float removeSpaceWidth = textPaint.measureText(removeSpaceText);
                    float spaceCount = trimSpaceText.length() - removeSpaceText.length();

                    float eachSpaceWidth = (textViewWidth - removeSpaceWidth) / spaceCount;

                    SpannableString spannableString = new SpannableString(lineString);
                    for (int j = 0; j < trimSpaceText.length(); j++) {
                        char c = trimSpaceText.charAt(j);
                        if (c == ' ') {
                            Drawable drawable = new ColorDrawable(0x00ffffff);
                            drawable.setBounds(0, 0, (int) eachSpaceWidth, 0);
                            ImageSpan span = new ImageSpan(drawable);
                            spannableString.setSpan(span, j, j + 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
                        }
                    }

                    builder.append(spannableString);
                }

                textView.setText(builder);
                isJustify.set(true);
            }
        }
    });
}

コードをGitHubに配置しました:https : //github.com/twiceyuan/TextJustification

概要:

概観


1
XMLプレビューでは機能しませんが、実際のデバイスでは
うまく

15

XMLレイアウト:TextViewではなくWebViewを宣言する

<WebView
 android:id="@+id/textContent"
 android:layout_width="fill_parent"
 android:layout_height="wrap_content" />

Javaコード:テキストデータをWebViewに設定

WebView view = (WebView) findViewById(R.id.textContent);
String text;
text = "<html><body><p align=\"justify\">";
text+= "This is the text will be justified when displayed!!!";
text+= "</p></body></html>";
view.loadData(text, "text/html", "utf-8");

これで問題が解決する場合があります。その完全に私のために働いた。


9

これが私のやり方です、私ができる最もエレガントな方法だと思います。このソリューションでは、レイアウトで行う必要があるのは次のことだけです。

  • 追加のxmlns宣言を追加する
  • あなたの変更TextView、新しい名前空間にアンドロイドからのソーステキストの名前空間を
  • あなたの置き換えTextView秒でx.y.z.JustifiedTextView

これがコードです。私の携帯電話(Galaxy Nexus Android 4.0.2、Galaxy Teos Android 2.1)では問題なく動作します。もちろん、私のパッケージ名をあなたのものに置き換えてもかまいません。

/assets/justified_textview.css

body {
    font-size: 1.0em;
    color: rgb(180,180,180);
    text-align: justify;
}

@media screen and (-webkit-device-pixel-ratio: 1.5) {
    /* CSS for high-density screens */
    body {
        font-size: 1.05em;
    }
}

@media screen and (-webkit-device-pixel-ratio: 2.0) {
    /* CSS for extra high-density screens */
    body {
        font-size: 1.1em;
    }
}

/res/values/attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="JustifiedTextView">
        <attr name="text" format="reference" />
    </declare-styleable>
</resources>

/res/layout/test.xml

<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:myapp="http://schemas.android.com/apk/res/net.bicou.myapp"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical" >

        <net.bicou.myapp.widget.JustifiedTextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            myapp:text="@string/surv1_1" />

    </LinearLayout>
</ScrollView>

/src/net/bicou/myapp/widget/JustifiedTextView.java

package net.bicou.myapp.widget;

import net.bicou.myapp.R;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Color;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.view.View;
import android.webkit.WebView;

public class JustifiedTextView extends WebView {
    public JustifiedTextView(final Context context) {
        this(context, null, 0);
    }

    public JustifiedTextView(final Context context, final AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public JustifiedTextView(final Context context, final AttributeSet attrs, final int defStyle) {
        super(context, attrs, defStyle);

        if (attrs != null) {
            final TypedValue tv = new TypedValue();
            final TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.JustifiedTextView, defStyle, 0);
            if (ta != null) {
                ta.getValue(R.styleable.JustifiedTextView_text, tv);

                if (tv.resourceId > 0) {
                    final String text = context.getString(tv.resourceId).replace("\n", "<br />");
                    loadDataWithBaseURL("file:///android_asset/",
                            "<html><head>" +
                                    "<link rel=\"stylesheet\" type=\"text/css\" href=\"justified_textview.css\" />" +
                                    "</head><body>" + text + "</body></html>",

                                    "text/html", "UTF8", null);
                    setTransparentBackground();
                }
            }
        }
    }

    public void setTransparentBackground() {
        try {
            setLayerType(View.LAYER_TYPE_SOFTWARE, null);
        } catch (final NoSuchMethodError e) {
        }

        setBackgroundColor(Color.TRANSPARENT);
        setBackgroundDrawable(null);
        setBackgroundResource(0);
    }
}

Android 3以降で透明な背景を取得するには、レンダリングをソフトウェアに設定する必要があります。したがって、古いバージョンのAndroidのtry-catchです。

お役に立てれば!

PS:期待される動作を得るために、これをAndroid 3以降のアクティビティ全体に追加すると役立つ場合があることに注意してください。
android:hardwareAccelerated="false"


これはwebViewベースのソリューションです。textviewがwebviewやscrollviewよりも軽いことを考えると、誰もがまだtextviewベースであることを発見しました。
スーパーユーザー



6

この問題を解決するために自分のクラスを作成します。ここでは、2つの引数を取る静的なjustify関数を呼び出す必要があります。

  1. テキストビューオブジェクト
  2. コンテンツの幅(テキストビューの合計幅)

//主な活動

package com.fawad.textjustification;
import android.app.Activity;
import android.database.Cursor;
import android.graphics.Point;
import android.graphics.Typeface;
import android.os.Bundle;
import android.util.DisplayMetrics;
import android.view.Display;
import android.view.Gravity;
import android.view.Menu;
import android.widget.TextView;

public class MainActivity extends Activity {
    static Point size;
    static float density;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Display display = getWindowManager().getDefaultDisplay();
        size=new Point();
        DisplayMetrics dm=new DisplayMetrics();
        display.getMetrics(dm);
        density=dm.density;
        display.getSize(size);


        TextView tv=(TextView)findViewById(R.id.textView1);
        Typeface typeface=Typeface.createFromAsset(this.getAssets(), "Roboto-Medium.ttf");
        tv.setTypeface(typeface);
        tv.setLineSpacing(0f, 1.2f);
        tv.setTextSize(10*MainActivity.density);

        //some random long text
         String myText=getResources().getString(R.string.my_text);

         tv.setText(myText);
        TextJustification.justify(tv,size.x);


    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

}

// TextJustificationClass

package com.fawad.textjustification;

import java.util.ArrayList;

import android.graphics.Paint;
import android.text.TextUtils;
import android.widget.TextView;

public class TextJustification {

    public static void justify(TextView textView,float contentWidth) {
        String text=textView.getText().toString();
        Paint paint=textView.getPaint();

        ArrayList<String> lineList=lineBreak(text,paint,contentWidth);

        textView.setText(TextUtils.join(" ", lineList).replaceFirst("\\s", ""));
    }


    private static ArrayList<String> lineBreak(String text,Paint paint,float contentWidth){
        String [] wordArray=text.split("\\s"); 
        ArrayList<String> lineList = new ArrayList<String>();
        String myText="";

        for(String word:wordArray){
            if(paint.measureText(myText+" "+word)<=contentWidth)
                myText=myText+" "+word;
            else{
                int totalSpacesToInsert=(int)((contentWidth-paint.measureText(myText))/paint.measureText(" "));
                lineList.add(justifyLine(myText,totalSpacesToInsert));
                myText=word;
            }
        }
        lineList.add(myText);
        return lineList;
    }

    private static String justifyLine(String text,int totalSpacesToInsert){
        String[] wordArray=text.split("\\s");
        String toAppend=" ";

        while((totalSpacesToInsert)>=(wordArray.length-1)){
            toAppend=toAppend+" ";
            totalSpacesToInsert=totalSpacesToInsert-(wordArray.length-1);
        }
        int i=0;
        String justifiedText="";
        for(String word:wordArray){
            if(i<totalSpacesToInsert)
                justifiedText=justifiedText+word+" "+toAppend;

            else                
                justifiedText=justifiedText+word+toAppend;

            i++;
        }

        return justifiedText;
    }

}

// XML

 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"

    tools:context=".MainActivity" 
    >



    <ScrollView
        android:id="@+id/scrollView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
         >

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="vertical"

             >
            <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/hello_world" />
        </LinearLayout>
    </ScrollView>

</RelativeLayout>

少なくとも "\ n"またはSystem.getProperty( "line.separator")を尊重するためにこの例を完成させてください:)
ceph3us

5

FILL_HORIZONTALと同等 CENTER_HORIZONTALです。このコードスニペットは、textviewのソースコードで確認できます。

case Gravity.CENTER_HORIZONTAL:
case Gravity.FILL_HORIZONTAL:
    return (mLayout.getLineWidth(0) - ((mRight - mLeft) -
            getCompoundPaddingLeft() - getCompoundPaddingRight())) /
            getHorizontalFadingEdgeLength();

4

この問題にはCustomViewがあり、このカスタムテキストビューはJustified Text Viewをサポートしています。

これで戦利品:JustifiedTextView

import java.util.ArrayList;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Typeface;
import android.text.TextPaint;
import android.view.View;

public class JustifiedTextView extends View {
        String text;
        ArrayList<Line> linesCollection = new ArrayList<Line>();
        TextPaint textPaint;
        Typeface font;
        int textColor;
        float textSize = 42f, lineHeight = 57f, wordSpacing = 15f, lineSpacing = 15f;
        float onBirim, w, h;
        float leftPadding, rightPadding;

        public JustifiedTextView(Context context, String text) {
                super(context);
                this.text = text;
                init();
        }

        private void init() {
                textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG);
                textColor = Color.BLACK;
        }

        @Override
        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
                super.onMeasure(widthMeasureSpec, heightMeasureSpec);

                if (font != null) {
                        font = Typeface.createFromAsset(getContext().getAssets(), "font/Trykker-Regular.ttf");
                        textPaint.setTypeface(font);
                }
                textPaint.setColor(textColor);

                int minw = getPaddingLeft() + getPaddingRight() + getSuggestedMinimumWidth();
                w = resolveSizeAndState(minw, widthMeasureSpec, 1);
                h = MeasureSpec.getSize(widthMeasureSpec);

                onBirim = 0.009259259f * w;
                lineHeight = textSize + lineSpacing;
                leftPadding = 3 * onBirim + getPaddingLeft();
                rightPadding = 3 * onBirim + getPaddingRight();

                textPaint.setTextSize(textSize);

                wordSpacing = 15f;
                Line lineBuffer = new Line();
                this.linesCollection.clear();
                String[] lines = text.split("\n");
                for (String line : lines) {
                        String[] words = line.split(" ");
                        lineBuffer = new Line();
                        float lineWidth = leftPadding + rightPadding;
                        float totalWordWidth = 0;
                        for (String word : words) {
                                float ww = textPaint.measureText(word) + wordSpacing;
                                if (lineWidth + ww + (lineBuffer.getWords().size() * wordSpacing) > w) {// is
                                        lineBuffer.addWord(word);
                                        totalWordWidth += textPaint.measureText(word);
                                        lineBuffer.setSpacing((w - totalWordWidth - leftPadding - rightPadding) / (lineBuffer.getWords().size() - 1));
                                        this.linesCollection.add(lineBuffer);
                                        lineBuffer = new Line();
                                        totalWordWidth = 0;
                                        lineWidth = leftPadding + rightPadding;
                                } else {
                                        lineBuffer.setSpacing(wordSpacing);
                                        lineBuffer.addWord(word);
                                        totalWordWidth += textPaint.measureText(word);
                                        lineWidth += ww;
                                }
                        }
                        this.linesCollection.add(lineBuffer);
                }
                setMeasuredDimension((int) w, (int) ((this.linesCollection.size() + 1) * lineHeight + (10 * onBirim)));
        }

        @Override
        protected void onDraw(Canvas canvas) {
                super.onDraw(canvas);
                canvas.drawLine(0f, 10f, getMeasuredWidth(), 10f, textPaint);
                float x, y = lineHeight + onBirim;
                for (Line line : linesCollection) {
                        x = leftPadding;
                        for (String s : line.getWords()) {
                                canvas.drawText(s, x, y, textPaint);
                                x += textPaint.measureText(s) + line.spacing;
                        }
                        y += lineHeight;
                }
        }

        public String getText() {
                return text;
        }

        public void setText(String text) {
                this.text = text;
        }

        public Typeface getFont() {
                return font;
        }

        public void setFont(Typeface font) {
                this.font = font;
        }

        public float getLineHeight() {
                return lineHeight;
        }

        public void setLineHeight(float lineHeight) {
                this.lineHeight = lineHeight;
        }

        public float getLeftPadding() {
                return leftPadding;
        }

        public void setLeftPadding(float leftPadding) {
                this.leftPadding = leftPadding;
        }

        public float getRightPadding() {
                return rightPadding;
        }

        public void setRightPadding(float rightPadding) {
                this.rightPadding = rightPadding;
        }

        public void setWordSpacing(float wordSpacing) {
                this.wordSpacing = wordSpacing;
        }

        public float getWordSpacing() {
                return wordSpacing;
        }

        public float getLineSpacing() {
                return lineSpacing;
        }

        public void setLineSpacing(float lineSpacing) {
                this.lineSpacing = lineSpacing;
        }

        class Line {
                ArrayList<String> words = new ArrayList<String>();
                float spacing = 15f;

                public Line() {
                }

                public Line(ArrayList<String> words, float spacing) {
                        this.words = words;
                        this.spacing = spacing;
                }

                public void setSpacing(float spacing) {
                        this.spacing = spacing;
                }

                public float getSpacing() {
                        return spacing;
                }

                public void addWord(String s) {
                        words.add(s);
                }

                public ArrayList<String> getWords() {
                        return words;
                }
        }
}

上記のクラスをsrcフォルダーに追加し、次のサンプルコードを使用してレイアウトに追加します。

JustifiedTextView jtv= new JustifiedTextView(getApplicationContext(), "Lorem ipsum dolor sit amet... ");
LinearLayout place = (LinearLayout) findViewById(R.id.book_profile_content);
place.addView(jtv);

4

githubでここ参照してください

プロジェクトに2つのファイル「TextJustifyUtils.java」と「TextViewEx.java」をインポートするだけです。

public class TextJustifyUtils {
    // Please use run(...) instead
    public static void justify(TextView textView) {
        Paint paint = new Paint();

        String[] blocks;
        float spaceOffset = 0;
        float textWrapWidth = 0;

        int spacesToSpread;
        float wrappedEdgeSpace;
        String block;
        String[] lineAsWords;
        String wrappedLine;
        String smb = "";
        Object[] wrappedObj;

        // Pull widget properties
        paint.setColor(textView.getCurrentTextColor());
        paint.setTypeface(textView.getTypeface());
        paint.setTextSize(textView.getTextSize());

        textWrapWidth = textView.getWidth();
        spaceOffset = paint.measureText(" ");
        blocks = textView.getText().toString().split("((?<=\n)|(?=\n))");

        if (textWrapWidth < 20) {
            return;
        }

        for (int i = 0; i < blocks.length; i++) {
            block = blocks[i];

            if (block.length() == 0) {
                continue;
            } else if (block.equals("\n")) {
                smb += block;
                continue;
            }

            block = block.trim();

            if (block.length() == 0)
                continue;

            wrappedObj = TextJustifyUtils.createWrappedLine(block, paint,
                    spaceOffset, textWrapWidth);
            wrappedLine = ((String) wrappedObj[0]);
            wrappedEdgeSpace = (Float) wrappedObj[1];
            lineAsWords = wrappedLine.split(" ");
            spacesToSpread = (int) (wrappedEdgeSpace != Float.MIN_VALUE ? wrappedEdgeSpace
                    / spaceOffset
                    : 0);

            for (String word : lineAsWords) {
                smb += word + " ";

                if (--spacesToSpread > 0) {
                    smb += " ";
                }
            }

            smb = smb.trim();

            if (blocks[i].length() > 0) {
                blocks[i] = blocks[i].substring(wrappedLine.length());

                if (blocks[i].length() > 0) {
                    smb += "\n";
                }

                i--;
            }
        }

        textView.setGravity(Gravity.LEFT);
        textView.setText(smb);
    }

    protected static Object[] createWrappedLine(String block, Paint paint,
            float spaceOffset, float maxWidth) {
        float cacheWidth = maxWidth;
        float origMaxWidth = maxWidth;

        String line = "";

        for (String word : block.split("\\s")) {
            cacheWidth = paint.measureText(word);
            maxWidth -= cacheWidth;

            if (maxWidth <= 0) {
                return new Object[] { line, maxWidth + cacheWidth + spaceOffset };
            }

            line += word + " ";
            maxWidth -= spaceOffset;

        }

        if (paint.measureText(block) <= origMaxWidth) {
            return new Object[] { block, Float.MIN_VALUE };
        }

        return new Object[] { line, maxWidth };
    }

    final static String SYSTEM_NEWLINE = "\n";
    final static float COMPLEXITY = 5.12f; // Reducing this will increase
                                            // efficiency but will decrease
                                            // effectiveness
    final static Paint p = new Paint();

    public static void run(final TextView tv, float origWidth) {
        String s = tv.getText().toString();
        p.setTypeface(tv.getTypeface());
        String[] splits = s.split(SYSTEM_NEWLINE);
        float width = origWidth - 5;
        for (int x = 0; x < splits.length; x++)
            if (p.measureText(splits[x]) > width) {
                splits[x] = wrap(splits[x], width, p);
                String[] microSplits = splits[x].split(SYSTEM_NEWLINE);
                for (int y = 0; y < microSplits.length - 1; y++)
                    microSplits[y] = justify(removeLast(microSplits[y], " "),
                            width, p);
                StringBuilder smb_internal = new StringBuilder();
                for (int z = 0; z < microSplits.length; z++)
                    smb_internal.append(microSplits[z]
                            + ((z + 1 < microSplits.length) ? SYSTEM_NEWLINE
                                    : ""));
                splits[x] = smb_internal.toString();
            }
        final StringBuilder smb = new StringBuilder();
        for (String cleaned : splits)
            smb.append(cleaned + SYSTEM_NEWLINE);
        tv.setGravity(Gravity.LEFT);
        tv.setText(smb);
    }

    private static String wrap(String s, float width, Paint p) {
        String[] str = s.split("\\s"); // regex
        StringBuilder smb = new StringBuilder(); // save memory
        smb.append(SYSTEM_NEWLINE);
        for (int x = 0; x < str.length; x++) {
            float length = p.measureText(str[x]);
            String[] pieces = smb.toString().split(SYSTEM_NEWLINE);
            try {
                if (p.measureText(pieces[pieces.length - 1]) + length > width)
                    smb.append(SYSTEM_NEWLINE);
            } catch (Exception e) {
            }
            smb.append(str[x] + " ");
        }
        return smb.toString().replaceFirst(SYSTEM_NEWLINE, "");
    }

    private static String removeLast(String s, String g) {
        if (s.contains(g)) {
            int index = s.lastIndexOf(g);
            int indexEnd = index + g.length();
            if (index == 0)
                return s.substring(1);
            else if (index == s.length() - 1)
                return s.substring(0, index);
            else
                return s.substring(0, index) + s.substring(indexEnd);
        }
        return s;
    }

    private static String justifyOperation(String s, float width, Paint p) {
        float holder = (float) (COMPLEXITY * Math.random());
        while (s.contains(Float.toString(holder)))
            holder = (float) (COMPLEXITY * Math.random());
        String holder_string = Float.toString(holder);
        float lessThan = width;
        int timeOut = 100;
        int current = 0;
        while (p.measureText(s) < lessThan && current < timeOut) {
            s = s.replaceFirst(" ([^" + holder_string + "])", " "
                    + holder_string + "$1");
            lessThan = p.measureText(holder_string) + lessThan
                    - p.measureText(" ");
            current++;
        }
        String cleaned = s.replaceAll(holder_string, " ");
        return cleaned;
    }

    private static String justify(String s, float width, Paint p) {
        while (p.measureText(s) < width) {
            s = justifyOperation(s, width, p);
        }
        return s;
    }
}

そして

public class TextViewEx extends TextView {
    private Paint paint = new Paint();

    private String[] blocks;
    private float spaceOffset = 0;
    private float horizontalOffset = 0;
    private float verticalOffset = 0;
    private float horizontalFontOffset = 0;
    private float dirtyRegionWidth = 0;
    private boolean wrapEnabled = false;
    int left, top, right, bottom = 0;
    private Align _align = Align.LEFT;
    private float strecthOffset;
    private float wrappedEdgeSpace;
    private String block;
    private String wrappedLine;
    private String[] lineAsWords;
    private Object[] wrappedObj;

    private Bitmap cache = null;
    private boolean cacheEnabled = false;

    public TextViewEx(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        // set a minimum of left and right padding so that the texts are not too
        // close to the side screen
        // this.setPadding(10, 0, 10, 0);
    }

    public TextViewEx(Context context, AttributeSet attrs) {
        super(context, attrs);
        // this.setPadding(10, 0, 10, 0);
    }

    public TextViewEx(Context context) {
        super(context);
        // this.setPadding(10, 0, 10, 0);
    }

    @Override
    public void setPadding(int left, int top, int right, int bottom) {
        // TODO Auto-generated method stub
        super.setPadding(left + 10, top, right + 10, bottom);
    }

    @Override
    public void setDrawingCacheEnabled(boolean cacheEnabled) {
        this.cacheEnabled = cacheEnabled;
    }

    public void setText(String st, boolean wrap) {
        wrapEnabled = wrap;
        super.setText(st);
    }

    public void setTextAlign(Align align) {
        _align = align;
    }

    @SuppressLint("NewApi")
    @Override
    protected void onDraw(Canvas canvas) {
        // If wrap is disabled then,
        // request original onDraw
        if (!wrapEnabled) {
            super.onDraw(canvas);
            return;
        }

        // Active canas needs to be set
        // based on cacheEnabled
        Canvas activeCanvas = null;

        // Set the active canvas based on
        // whether cache is enabled
        if (cacheEnabled) {

            if (cache != null) {
                // Draw to the OS provided canvas
                // if the cache is not empty
                canvas.drawBitmap(cache, 0, 0, paint);
                return;
            } else {
                // Create a bitmap and set the activeCanvas
                // to the one derived from the bitmap
                cache = Bitmap.createBitmap(getWidth(), getHeight(),
                        Config.ARGB_4444);
                activeCanvas = new Canvas(cache);
            }
        } else {
            // Active canvas is the OS
            // provided canvas
            activeCanvas = canvas;
        }

        // Pull widget properties
        paint.setColor(getCurrentTextColor());
        paint.setTypeface(getTypeface());
        paint.setTextSize(getTextSize());
        paint.setTextAlign(_align);
        paint.setFlags(Paint.ANTI_ALIAS_FLAG);

        // minus out the paddings pixel
        dirtyRegionWidth = getWidth() - getPaddingLeft() - getPaddingRight();
        int maxLines = Integer.MAX_VALUE;
        int currentapiVersion = android.os.Build.VERSION.SDK_INT;
        if (currentapiVersion >= android.os.Build.VERSION_CODES.JELLY_BEAN) {
            maxLines = getMaxLines();
        }
        int lines = 1;
        blocks = getText().toString().split("((?<=\n)|(?=\n))");
        verticalOffset = horizontalFontOffset = getLineHeight() - 0.5f; // Temp
                                                                        // fix
        spaceOffset = paint.measureText(" ");

        for (int i = 0; i < blocks.length && lines <= maxLines; i++) {
            block = blocks[i];
            horizontalOffset = 0;

            if (block.length() == 0) {
                continue;
            } else if (block.equals("\n")) {
                verticalOffset += horizontalFontOffset;
                continue;
            }

            block = block.trim();

            if (block.length() == 0) {
                continue;
            }

            wrappedObj = TextJustifyUtils.createWrappedLine(block, paint,
                    spaceOffset, dirtyRegionWidth);

            wrappedLine = ((String) wrappedObj[0]);
            wrappedEdgeSpace = (Float) wrappedObj[1];
            lineAsWords = wrappedLine.split(" ");
            strecthOffset = wrappedEdgeSpace != Float.MIN_VALUE ? wrappedEdgeSpace
                    / (lineAsWords.length - 1)
                    : 0;

            for (int j = 0; j < lineAsWords.length; j++) {
                String word = lineAsWords[j];
                if (lines == maxLines && j == lineAsWords.length - 1) {
                    activeCanvas.drawText("...", horizontalOffset,
                            verticalOffset, paint);

                } else if (j == 0) {
                    // if it is the first word of the line, text will be drawn
                    // starting from right edge of textview
                    if (_align == Align.RIGHT) {
                        activeCanvas.drawText(word, getWidth()
                                - (getPaddingRight()), verticalOffset, paint);
                        // add in the paddings to the horizontalOffset
                        horizontalOffset += getWidth() - (getPaddingRight());
                    } else {
                        activeCanvas.drawText(word, getPaddingLeft(),
                                verticalOffset, paint);
                        horizontalOffset += getPaddingLeft();
                    }

                } else {
                    activeCanvas.drawText(word, horizontalOffset,
                            verticalOffset, paint);
                }
                if (_align == Align.RIGHT)
                    horizontalOffset -= paint.measureText(word) + spaceOffset
                            + strecthOffset;
                else
                    horizontalOffset += paint.measureText(word) + spaceOffset
                            + strecthOffset;
            }

            lines++;

            if (blocks[i].length() > 0) {
                blocks[i] = blocks[i].substring(wrappedLine.length());
                verticalOffset += blocks[i].length() > 0 ? horizontalFontOffset
                        : 0;
                i--;
            }
        }

        if (cacheEnabled) {
            // Draw the cache onto the OS provided
            // canvas.
            canvas.drawBitmap(cache, 0, 0, paint);
        }
    }
}

今、あなたが通常のtextViewを使う場合:

<TextView
                android:id="@+id/original"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="@string/lorum_ipsum" />

単に使う

<yourpackagename.TextViewEx
                android:id="@+id/changed"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="@string/lorum_ipsum" />

変数を定義し、正当化をtrueに設定します。

TextViewEx changed = (TextViewEx) findViewById(R.id.changed);
changed.setText(getResources().getString(R.string.lorum_ipsum),true);

太字のテキストが機能しません。何か修正がある場合は、助けてください
praveenb

4

TextView XMLのAndroid Text Justify

単にAndroidでXMLを使用してテキストを正当化します。あなたは単にtextviewウィジェットに実装することができます。

 <TextView
    android:justificationMode="inter_word"
/>

デフォルトは android:justificationMode="none"


2

2つのオプションがあると思います。

  • NDKを介してこれに特化したPangoのようなものを使用し、OpenGLまたはその他のサーフェスにテキストをレンダリングします。

  • Paint.measureText()などを使用して単語の長さを取得し、カスタムビューのキャンバスに手動で配置します。


2

Androidでは、テキストを左揃えにし、背景色が切り捨てられないようにするには、これを試してください、それは私にとってはうまくいき、Android、FF、IE、Chromeで一貫した結果を生成しますが、テキストの間に残っているスペースを測定する必要がありますパディングを計算するとき。

<td style="font-family:Calibri,Arial;
    font-size:15px;
    font-weight:800;
    background-color:#f5d5fd;
    color:black;
    border-style:solid;
    border-width:1px;
    border-color:#bd07eb;
    padding-left:10px;
    padding-right:1000px;
    padding-top:3px;
    padding-bottom:3px;
>

ハックとは、padding-right:1000px;テキストを左端に押し出すハックです。

CSSまたはHTMLで左にコードを配置したり、コードを正当化したりすると、背景の幅が半分になります。



1

Androidはまだ完全な正当化をサポートしていません。textviewを使用する代わりに、Webviewを使用してHTMLを正当化できます。それはとてもうまくいきます。はっきりしない場合は、お気軽に私に聞いてください。


それができます。ことができますが、私たちはの背景を設定しますWebView transparent。背景画像があります。
Mr.India 2013

これは記憶力の問題だとは思わない。
superUser


1

TextViewコンテンツの正当化: 簡単な人たちは、TextViewタグ内でandroid:justificationMode = "inter_word"を使用するだけです。

 <TextView
    android:id="@+id/textView2"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_marginTop="92dp"
    android:text="@string/contents"
    android:layout_margin="20dp"
    android:justificationMode="inter_word"
     />

-4

<を使用してみてRelativeLayout >(fill_parentを確実にしてください)、次に追加android:layout_alignParentLeft="true"して

android:layout_alignParentRight="true" LEFT&RIGHTの外側に配置したい要素に。

BLAM、正当化された!



3
それでも彼が探しているものではありません。WikipediaのJustificationを参照してください:en.wikipedia.org/wiki/Justification_(typesetting)
Kevin Coppock

正当な理由ではありません
アラシュハタミ2018

-5

設定する必要があります

android:layout_height="wrap_content"

そして

android:layout_centerInParent="true"

11
これは、完全な正当化されていないテキストを中央揃えにします
Janusz 2013年

-12

これは本当にあなたのテキストを正当化しませんが、

android:gravity="center_horizontal"

あなたが持っている最良の選択です。


9
いいえ、それはテキストを中央揃えにします。それはそれを正当化しません。ウィキペディアの引用:「正当化されたテキストでは、単語間のスペース、および程度は低いがグリフまたは文字間のスペース(カーニング)が、テキストを左右両方のマージンに揃えるために引き伸ばされるか、時々圧縮されます。」
CommonsWare 2012

テキストはあなたのコードによって正当化されませんが、テキストを中央に水平に配置します
Matteo
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.