我有一个简单的LinearLayout
,只包含扩展了TextView
的自定义视图(我将开始称之为“IdiomView
”)和ListView
。 IdiomView
与普通TextView
的唯一区别是,我重写了onDraw()
方法来迭代地减小文本大小,直到文本长度少于3行。我的问题是,绘制视图时,用户将看到以下内容:
______________
|__ACTION_BAR__|
| IdiomView |
|______________|
| |
| ListView |
| |
| |
|______________|
很快变成:
______________
|__ACTION_BAR__|
|__IdiomView __|
| |
| ListView |
| |
| |
| |
|______________|
即,绘制了ListView,然后在
IdiomView
整理出其大小之后跳了起来。我想要的是一种在绘制ListView之前等待
IdiomView
完全绘制的方法。这篇What event is fired after all views are fully drawn?帖子解释了在绘制完成后如何通过调用View.post(Runnable)
来排队线程。问题是我重写的onDraw()
方法多次调用onDraw()
来计算较小的文本是否覆盖不到3行,因此在我希望ListView
出现之前,此元素可能“完成绘制”多次。我感谢所有评论和答案。这是我当前的代码:
布局XML:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/main_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_margin="10dp"
android:background="@color/off_white"
android:orientation="vertical" >
<carter.cwords.idioms.IdiomView
android:id="@+id/idiom"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_marginBottom="10dp"
android:textColor="@color/transparent"
android:textSize="28sp"
android:textStyle="italic"
android:visibility="invisible" />
<ListView
android:id="@+id/quote_list"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:choiceMode="none"
android:footerDividersEnabled="false"
android:headerDividersEnabled="false"
android:visibility="invisible" />
</LinearLayout>
Activity :
private IdiomView mIdiomTextView;
private ListView mQuoteList;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.idiom_of_the_day);
mIdiomTextView = (IdiomView) findViewById(R.id.idiom);
mQuoteList = (ListView) findViewById(R.id.quote_list);
// Populate page data onResume()
}
@Override
protected void onResume() {
super.onResume();
sendRequest(R.string.url_idiom_of_the_day, new AfterRequest(){
@Override
public void useResults(Document resultXml) {
if(resultXml != null){
Log.i(getClass().getSimpleName(), "useResults()");
String idiomString = XmlUtilities.getTextValue(resultXml, NetworkHelper.XML_TAG_IDIOM_CONTENT);
logDebug("idiomString: " + idiomString);
mIdiomTextView.setText("\"" + idiomString + "\"");
mQuoteList.setAdapter(new ContentAdapter(mContext, resultXml));
mIdiomTextView.setVisibility(View.VISIBLE);
mIdiomTextView.post(new Runnable(){
@Override
public void run() {
mQuoteList.setVisibility(View.VISIBLE);
}
});
}
}
});
}
IdiomView
:public class IdiomView extends TextView {
public IdiomView(Context context) {
super(context);
}
public IdiomView(Context context, AttributeSet attrs){
super(context, attrs);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Log.i(getClass().getSimpleName(), "onDraw(): " + this.getLineCount());
if(this.getLineCount() > 2){
this.setTextSize(TypedValue.COMPLEX_UNIT_PX, this.getTextSize()-1);
}
else{
this.setTextColor(getResources().getColor(R.color.text));
}
Log.i(getClass().getSimpleName(), "onDraw(): " + this.getLineCount());
}
}
非常感谢你。
最佳答案
重复设置文本大小直到找到合适的字体大小可能会非常低效,因为您的TextView
每次都需要重新测量和重绘。代替此操作,您只能使用TextView.getPaint()
方法测量和设置文本一次。然后,您可以循环使用Paint
提供不同的字体大小来测量文本,直到Paint.measureText()
返回的值小于小部件当前宽度的两倍。