我有一個簡單的LinearLayout
只包含一個自定義視圖,延伸TextView
(我將開始調用「IdiomView
」)和一個ListView
。與正常的TextView
相比,IdiomView
唯一的區別在於,我已經重寫了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>
活動:
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());
}
}
非常感謝你。
通過使用getPaint()方法,可以避免重複設置文本大小(同時重新繪製小部件)。使用該「Paint」來測量文本在循環中的寬度(使用不同的文本大小),直到您看到兩次將適合可用的最大寬度(對於TextView)的值,並將該直接值用於文本大小 – Luksprog 2013-02-20 21:35:20
感謝Luksprog。這就是我應該如何處理這個問題。 [This thread](http://stackoverflow.com/questions/14276853/how-to-measure-textview-height-based-on-device-width-and-font-size)提供了有關測量TextView的詳細信息。 – 2013-02-21 23:42:21