OSCHINA客戶端完全剖析(四)動彈

動彈1
動彈2

跟前篇完全一樣的方法追蹤源碼進行分析。而在這之前也不妨先進行一下預估:UI呈現上與綜合是相同的,僅在單個數據項上多出了左側的圓形頭像以及右下角的評論數展示,還有就是動彈的內容也是可以顯示圖片的。所以,可能還會看到相應的圖片加載與緩存處理。

TweetsViewPagerFragment extends BaseViewPagerFragment

/**
 * 動彈界面(包括最新動彈、熱門動彈、我的動彈)
 * 
 * @author FireAnt(http://my.oschina.net/LittleDY)
 * @created 2014年9月25日 下午2:21:52
 * 
 */
public class TweetsViewPagerFragment extends BaseViewPagerFragment

BaseViewPagerFragment在之前篇章中已分析過,所以略去。
這里覆蓋的關鍵方法:

    @Override
    protected void onSetupTabAdapter(ViewPageFragmentAdapter adapter) {

        String[] title = getResources().getStringArray(
                R.array.tweets_viewpage_arrays);
        adapter.addTab(title[0], "new_tweets", TweetsFragment.class,
                getBundle(TweetsList.CATALOG_LATEST));
        adapter.addTab(title[1], "hot_tweets", TweetsFragment.class,
                getBundle(TweetsList.CATALOG_HOT));
        adapter.addTab(title[2], "my_tweets", TweetsFragment.class,
                getBundle(TweetsList.CATALOG_ME));
    }

TweetsFragment extends BaseListFragment<Tweet>

應該說這個類的結構也跟之前分析綜合時所見是完全雷同的,所不同的只在于多了長按、刪除等功能的支持。
直接看實現的Adapter:

    @Override
    protected TweetAdapter getListAdapter() {
        return new TweetAdapter();
    }

ListBaseAdapter<T extends Entity> extends BaseAdapter

提供多種狀態:

public static final int STATE_EMPTY_ITEM = 0;
    public static final int STATE_LOAD_MORE = 1;
    public static final int STATE_NO_MORE = 2;
    public static final int STATE_NO_DATA = 3;
    public static final int STATE_LESS_ONE_PAGE = 4;
    public static final int STATE_NETWORK_ERROR = 5;
    public static final int STATE_OTHER = 6;

數據集采用protected ArrayList<T> mDatas = new ArrayList<T>();并封裝相應操作如:

    public int getDataSizePlus1(){
        if(hasFooterView()){
            return getDataSize() + 1;
        }
        return getDataSize();
    }

    public int getDataSize() {
        return mDatas.size();
    }

FooterView操作:

    public void setFooterViewLoading(String loadMsg) {
        ProgressBar progress = (ProgressBar) mFooterView
                .findViewById(R.id.progressbar);
        TextView text = (TextView) mFooterView.findViewById(R.id.text);
        mFooterView.setVisibility(View.VISIBLE);
        progress.setVisibility(View.VISIBLE);
        text.setVisibility(View.VISIBLE);
        if (StringUtils.isEmpty(loadMsg)) {
            text.setText(_loadmoreText);
        } else {
            text.setText(loadMsg);
        }
    }

getView()中switch case不同的狀態,最末返回getRealView(position, convertView, parent);,而這是留給子類實現的,寫成抽象方法會更好一些:

protected View getRealView(int position, View convertView, ViewGroup parent) {
        return null;
    }

TweetAdapter extends ListBaseAdapter<Tweet>

這個recordBitmap一眼看上去并不明白是用來干什么的:

    private Bitmap recordBitmap;
    private Context context;

看關鍵方法getRealView的實現。
代碼很長就不貼全了,大意是先ViewHolder的使用,再是各項賦值控件顯隱什么的慣常操作:

        vh.face.setUserInfo(tweet.getAuthorid(), tweet.getAuthor());
        vh.face.setAvatarUrl(tweet.getPortrait());
        vh.author.setText(tweet.getAuthor());
        vh.time.setText(StringUtils.friendly_time(tweet.getPubDate()));
        vh.content.setMovementMethod(MyLinkMovementMethod.a());
        vh.content.setFocusable(false);
        vh.content.setDispatchToParent(true);
        vh.content.setLongClickable(false);

查布局知這里的vh.face為一個自定義控件,保存用戶信息及顯示——包含圓形頭像。

粗略看看它的實現,CircleImageView extends ImageView:

    private static final ScaleType SCALE_TYPE = ScaleType.CENTER_CROP;

    private static final Bitmap.Config BITMAP_CONFIG = Bitmap.Config.ARGB_8888;
    private static final int COLORDRAWABLE_DIMENSION = 1;

    private static final int DEFAULT_BORDER_WIDTH = 0;
    private static final int DEFAULT_BORDER_COLOR = Color.BLACK;

可以看到它使用了置中放縮,色彩格式上使用了最大的ARGB_8888,依據情況也可以考慮換成ARGB_4444之類。另外從最后兩項來看,這個自定義控件應該是設置邊框寬度和色彩的。默認寬度為0,色彩為黑。

看CircleImageView#onDraw:

    @Override
    protected void onDraw(Canvas canvas) {
        if(!isDisplayCircle) {
            super.onDraw(canvas);
            return;
        }
        if (getDrawable() == null) {
            return;
        }

        canvas.drawCircle(getWidth() / 2, getHeight() / 2, mDrawableRadius, mBitmapPaint);
        if(mBorderWidth != 0){
          canvas.drawCircle(getWidth() / 2, getHeight() / 2, mBorderRadius, mBorderPaint);
        }
    }

畫筆初始化,可以看到它是使用了BitmapShader來實現圓形繪制的:

        mBitmapShader = new BitmapShader(mBitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);

        mBitmapPaint.setAntiAlias(true);
        mBitmapPaint.setShader(mBitmapShader);

另一種實現可以參考這篇鴻洋的博文

接著看圖片加載vh.face.setAvatarUrl(tweet.getPortrait());

    public void setAvatarUrl(String url) {
        if (StringUtils.isEmpty(url)) {
            setImageResource(R.drawable.widget_dface);
            return;
        }
        // 由于頭像地址默認加了一段參數需要去掉
        int end = url.indexOf('?');
        final String headUrl;
        if (end > 0) {
            headUrl = url.substring(0, end);
        } else {
            headUrl = url;
        }

        Core.getKJBitmap().display(this, headUrl, R.drawable.widget_dface, 0, 0,
                new BitmapCallBack() {
                    @Override
                    public void onFailure(Exception e) {
                        super.onFailure(e);
                        aty.runOnUiThread(new Runnable() {
                            @Override
                            public void run() {
                                setImageResource(R.drawable.widget_dface);
                            }
                        });
                        setImageResource(R.drawable.widget_dface);
                    }
                });
    }

onFailure寫得有點奇怪,最后的setImageResource(R.drawable.widget_dface);應該是不需要的。
可以看到,這里加載使用了其自家的開發框架KJFrameForAndroid,具體資料可以另行查看,所以再往下分析就沒有必要了。


動彈詳情

這與之前分析過的資訊詳情基本是一個套路,這里就不贅述了,實現上也是使用一個WebView進行呈現,最主要的差異應該是體現在填充方法:

    /**
     * 填充webview內容
     */
    private void fillWebViewBody() {
        StringBuffer body = new StringBuffer();
        body.append(ThemeSwitchUtils.getWebViewBodyString());
        body.append(UIHelper.WEB_STYLE + UIHelper.WEB_LOAD_IMAGES);

        StringBuilder tweetbody = new StringBuilder(mTweet.getBody());

        String tweetBody = TextUtils.isEmpty(mTweet.getImgSmall()) ? tweetbody
                .toString() : tweetbody.toString() + "<br/><img src=\""
                + mTweet.getImgSmall() + "\">";
        body.append(setHtmlCotentSupportImagePreview(tweetBody));

        UIHelper.addWebImageShow(getActivity(), mContent);
        // 封尾
        body.append("</div></body>");
        mContent.loadDataWithBaseURL(null, body.toString(), "text/html",
                "utf-8", null);
    }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,967評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,273評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,870評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,742評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,527評論 6 407
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,010評論 1 322
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,108評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,250評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,769評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,656評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,853評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,371評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,103評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,472評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,717評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,487評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,815評論 2 372

推薦閱讀更多精彩內容