Android仿簡(jiǎn)書(shū)長(zhǎng)按文章生成圖片效果

前言

使用簡(jiǎn)書(shū)APP的同學(xué)都知道戒良,簡(jiǎn)書(shū)有這樣一個(gè)功能:文章頁(yè)長(zhǎng)按內(nèi)容時(shí)底部會(huì)出現(xiàn)一個(gè) 生成圖片分享 的按鈕寂玲,點(diǎn)擊之后就可以將當(dāng)前的文章生成一張長(zhǎng)圖片塔插,這張圖片可以保存到本地或分享給好友,同時(shí)還可為圖片設(shè)置成為白和黑兩種風(fēng)格拓哟,很有藝術(shù)范想许。個(gè)人一直很喜歡這個(gè)功能。

但是從某一個(gè)版本開(kāi)始彰檬,這個(gè)功能開(kāi)始有bug了伸刃,生成的圖片只有底部的固定標(biāo)題谎砾,而沒(méi)有文章內(nèi)容逢倍,長(zhǎng)圖也變成了小短圖。向簡(jiǎn)書(shū)意見(jiàn)反饋后景图,得到的回復(fù)是较雕,使用點(diǎn)擊分享按鈕生成圖片功能;分享菜單包含的生成長(zhǎng)圖功能的確是可以的。但是亮蒋,還是很懷念之前長(zhǎng)按生成圖片的功能扣典,所以作為一名程序猿;懷著好奇的心情慎玖,決定自己去實(shí)現(xiàn)這樣一個(gè)功能.

效果預(yù)覽

老規(guī)矩贮尖,首先看一下實(shí)現(xiàn)后的效果;雖然整體沒(méi)有簡(jiǎn)書(shū)有范趁怔,個(gè)人感覺(jué)還是挺像的湿硝。

文章頁(yè)實(shí)現(xiàn)

內(nèi)容

文章頁(yè)內(nèi)容的實(shí)現(xiàn),沒(méi)有什么難點(diǎn)润努。布局總的來(lái)說(shuō)很簡(jiǎn)單关斜,包含戶信息和文章信息的一個(gè)LinearLayout,外加一個(gè)WebView即可铺浇。數(shù)據(jù)是根據(jù)布局中所需的內(nèi)容痢畜,封裝了一個(gè)HtmlBean 對(duì)象,而這個(gè)對(duì)象的則是通過(guò)使用Jsoup 解析當(dāng)前頁(yè)面的HTML文檔內(nèi)容獲得(這里使用Jsoup 方式獲取簡(jiǎn)書(shū)網(wǎng)頁(yè)內(nèi)容鳍侣,只是個(gè)人學(xué)習(xí)丁稀,沒(méi)有其他用意)。具體實(shí)現(xiàn)可查看 源碼

長(zhǎng)按菜單實(shí)現(xiàn)

這里特意說(shuō)一下倚聚,長(zhǎng)按彈出底部按鈕的實(shí)現(xiàn)方式二驰。一般情況下對(duì)于長(zhǎng)按效果的實(shí)現(xiàn),我們都會(huì)通過(guò)設(shè)置View的OnLongClickListene事件去實(shí)現(xiàn)相應(yīng)的功能秉沼,但是對(duì)于這里的WebView可以如下實(shí)現(xiàn):

mWebView.setOnCreateContextMenuListener(new View.OnCreateContextMenuListener() {

    @Override

    public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) {

        genImg.setVisibility(View.VISIBLE);

        T.showSToast(mContext, "再次點(diǎn)擊文章可隱藏圖片分享");

    }

});

// 點(diǎn)擊隱藏底部按鈕

mWebView.setOnTouchListener(new View.OnTouchListener() {

    @Override

    public boolean onTouch(View v, MotionEvent event) {

        switch (event.getAction()) {

            case MotionEvent.ACTION_DOWN:

                lastTime = SystemClock.uptimeMillis();

            break;

            case MotionEvent.ACTION_UP:

                if (SystemClock.uptimeMillis() - lastTime < 300) {

                    genImg.setVisibility(View.GONE);

                }

            break;

        }

        return false;

    }

});

這里通過(guò)監(jiān)聽(tīng)WebView的ContextMenu 監(jiān)聽(tīng)何時(shí)顯示底部按鈕桶雀;同時(shí)在onTouch方法中隱藏底部按鈕。

genImg.setOnClickListener(new View.OnClickListener() {

    @Override

    public void onClick(View v) {

        genImg.setVisibility(View.INVISIBLE);

        Intent intent = new Intent(FakeJianShuActivity.this, GenScreenShotActivity.class);

        intent.putExtra("data", mHtmlBean);

        startActivity(intent);

    }

});

點(diǎn)擊底部的Button就會(huì)跳轉(zhuǎn)到生成長(zhǎng)圖的界面唬复,同時(shí)將之前獲取到的HTMLBean對(duì)象傳遞過(guò)去矗积。

長(zhǎng)圖效果實(shí)現(xiàn)

這里首先說(shuō)一下實(shí)現(xiàn)思路(思路來(lái)源于 此 )。

  • 首先通過(guò)WebView加載一個(gè)本地的Html頁(yè)面敞咧,這個(gè)頁(yè)面包含一些固定棘捣,定義了一些標(biāo)簽。然后根據(jù)傳遞過(guò)來(lái)的mHtmlBean 對(duì)象中的信息休建,通過(guò)執(zhí)行JavaScript動(dòng)態(tài)的替換靜態(tài)HTML頁(yè)面中的內(nèi)容乍恐;

  • 關(guān)于黑白兩種風(fēng)格的實(shí)現(xiàn),同樣是WebView執(zhí)行Js测砂,動(dòng)態(tài)替換HTML中CSS 樣式茵烈,修改WebView的背景色呈現(xiàn)出兩種不同的UI 效果。

  • 通過(guò)WebView的capturePicture 和Canvas 可以生成出當(dāng)前WebView的Bitmap對(duì)象砌些,有了這個(gè)Bitmap就可以圖片保存的功能了呜投。

好了加匈,下面就通過(guò)代碼分別實(shí)現(xiàn)上述步驟。

Html 頁(yè)面

style="position:absolute;top: 0px;left: 12px;margin-bottom: 15px;"/>

    function changeContent(content) {

    document.getElementById('content').innerHTML = content;

}

這個(gè)HTML頁(yè)面的內(nèi)容很簡(jiǎn)單仑荐,在整個(gè)文檔左上角放置了一個(gè)小角標(biāo)雕拼,就是簡(jiǎn)書(shū)APP生成長(zhǎng)圖時(shí)的那個(gè)mark.

同時(shí)定義了一個(gè)JavaScript 方法,功能也很簡(jiǎn)單粘招,就是用傳遞的參數(shù)content替換article標(biāo)簽中的文檔內(nèi)容啥寇。

自定義WebView

為了方便,我們自定義WebView洒扎,這里看一下核心邏輯:

public class FakeWebView extends WebView {

 private boolean isFirstLoad = false;

    public void loadData(HtmlBean bean) {
        assembleData(bean);
        if (Build.VERSION.SDK_INT >= 21) {
            isFirstLoad = true;
            webView.setWebChromeClient(new WebChromeClient() {
                @Override
                public void onProgressChanged(WebView view, int newProgress) {
                    if (newProgress == 100) {
                        if (isFirstLoad) {
                            isFirstLoad = false;
                            Log.e("TAG", "onProgressChanged");
                            updateView();
                        }
                    }
                }
            });
        } else {
            isFirstLoad = true;
            webView.setVisibility(View.INVISIBLE);
            webView.setWebChromeClient(new WebChromeClient() {
                @Override
                public void onProgressChanged(WebView view, int newProgress) {
                    if (newProgress == 100) {
                        updateView();
                    if (!isFirstLoad)
                        webView.setVisibility(View.VISIBLE);
                    }
                }
            });
        }

        webView.loadUrl("file:///android_asset/JianShu.html");

    }

    private void assembleData(HtmlBean bean) {
        final String data = bean.getContent();
        final String title = bean.getTitle();
        final String username = bean.getUsername();
        final String publishTime = bean.getPublishTime();
        String title = "<h2>"+title+"</h2>";
        String Footer ="<p>"+username+"</p><p>"+publishTime+"</p>";
        content = Title + data + Footer;
    }

    public void updateView() {
        if (mode == MODE_DAY) {
            webView.setBackgroundColor(Color.WHITE);
        } else {
            webView.setBackgroundColor(Color.parseColor("#263238"));
            content = "
            " + content + "
            ";
        }

        webView.loadUrl(
            "javascript:changeContent(\"" + content.replace("\n", "\\n").replace("\"", "\\\"").replace("'", "\\'") + "\")");

    }

}

這幾個(gè)方法是生成長(zhǎng)圖最核心的方法示姿。在loadData 方法中首先調(diào)用了assembleData,這個(gè)方法會(huì)根據(jù)mHtmlBean 這個(gè)對(duì)象中的數(shù)據(jù)拼接出一段 HTML 文檔逊笆。在webView的loadUrl 方法中會(huì)從本地加載之前定義好的JianShu.html這個(gè)頁(yè)面栈戳。然后在頁(yè)面加載完成,即onProgressChanged 回調(diào)方法中newProgress 的值等于100時(shí)調(diào)用updateView方法难裆;這個(gè)方法會(huì)根據(jù)當(dāng)前設(shè)置的模式子檀,設(shè)置WebView的背景,如果是夜間模式乃戈,則會(huì)對(duì)assembleData 中生成的文檔外部在添加 一個(gè)灰色風(fēng)格的div標(biāo)簽褂痰,將整個(gè)內(nèi)容包在這個(gè)div標(biāo)簽中,最后WebView執(zhí)行JS方法 changeContent症虑,傳遞的參數(shù)就是之前我們拼接好的內(nèi)容缩歪。這樣整個(gè)WebView又會(huì)刷新一次,整個(gè)WebView的內(nèi)容就是文章內(nèi)容了谍憔。

GenScreenShotActivity代碼如下:

mFakeWebView = (FakeWebView) findViewById(R.id.fakeWebView);

bean = (HtmlBean) getIntent().getSerializableExtra("data");

RadioGroup changeMode = (RadioGroup) findViewById(R.id.changeMode);

changeMode.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {

    @Override

    public void onCheckedChanged(RadioGroup group, @IdRes int checkedId) {

        if (checkedId == R.id.rb_day) {

            mFakeWebView.setMode(FakeWebView.MODE_DAY);

        } else {

            mFakeWebView.setMode(FakeWebView.MODE_NIGHT);

        }

    }

});

mFakeWebView.loadData(bean);


public void setMode(@ViewMode int mode) {

    this.mode = mode;

    updateView();

}

這樣在Activity中匪蝙,mFakeWebView對(duì)象通過(guò)上一個(gè)頁(yè)面(文章頁(yè))傳遞的mHtmlBean 對(duì)象就可以更新當(dāng)前視圖了,同時(shí)可以通過(guò)RadioButton實(shí)現(xiàn)頁(yè)面風(fēng)格的切換习贫。

保存圖片

距離我們最后的目標(biāo) 生成長(zhǎng)圖片 逛球,前面的工作可以說(shuō)只是完成了50%,因?yàn)榈侥壳盀橹刮覀冎徊贿^(guò)是在WebView中把整個(gè)文章內(nèi)容加載出來(lái)而已苫昌;長(zhǎng)圖還沒(méi)有呢颤绕。因此,下面的工作就是通過(guò)WebView 生成長(zhǎng)圖祟身。

public Bitmap getScreenView(){
    Picture snapShot = webView.capturePicture();
    Bitmap bmp = Bitmap.createBitmap(snapShot.getWidth(),snapShot.getHeight(),     
                                         Bitmap.Config.ARGB_8888);
    Canvas canvas = new Canvas(bmp);
    snapShot.draw(canvas);
    return bmp;

}

WebVeiw 很人性化奥务,通過(guò)這個(gè)方法,我們就可以獲得當(dāng)前WebView視圖 可見(jiàn)與不可見(jiàn) 部分的Bitmap了袜硫。

其實(shí)通過(guò)WebView生成圖片并不是一件難事氯葬,難得是如何把我們這里的圖片保存下來(lái);因?yàn)槲覀冞@里生成的是長(zhǎng)圖父款,如下圖所示溢谤,這張照片的高度達(dá)到了驚人的。因此這里就要需要之前在 Bitmap 初探 中提到的第一種壓縮方法進(jìn)行文件大小的壓縮了憨攒。具體實(shí)現(xiàn)世杀,就不再重復(fù)貼出代碼了,有興趣的同學(xué)可參考 Github源碼 肝集。

到這里瞻坝,我們就完全實(shí)現(xiàn)了仿照簡(jiǎn)書(shū)長(zhǎng)按生成圖片的功能。那么回過(guò)頭再來(lái)看杏瞻,這樣一個(gè)功能所刀,為什么在我的手機(jī)上,簡(jiǎn)書(shū)APP的長(zhǎng)按功能會(huì)有bug呢捞挥。


缺陷

文章詳情頁(yè)的WebView是系統(tǒng)自帶的WebView浮创,在加載帶 代碼的文章時(shí),沒(méi)有對(duì)代碼類的內(nèi)容做特殊的解析砌函,因此無(wú)法對(duì)代碼高亮顯示斩披。只是最為普通的文本進(jìn)行了顯示,因此生成的長(zhǎng)圖中代碼也是普通文本讹俊。簡(jiǎn)書(shū)APP還是高大上呀垦沉,對(duì)代碼的高亮顯示正是棒棒噠!


后話

一個(gè)偶然的機(jī)會(huì)仍劈,在嘗試簡(jiǎn)書(shū)長(zhǎng)按生成圖片的功能時(shí)發(fā)現(xiàn)厕倍,原來(lái)簡(jiǎn)書(shū)是通過(guò)WebView選擇的區(qū)域生成第二頁(yè)的內(nèi)容;因此當(dāng)我在文章頁(yè)空白區(qū)域長(zhǎng)按后贩疙,點(diǎn)擊生成圖片時(shí)必然是只有空白的讹弯,只有底部的一些固定標(biāo)簽。因此这溅,這應(yīng)該不算是一個(gè)bug闸婴,只是為大家提供了一種更方便的功能,可以按自己喜歡的內(nèi)容生成更有效的長(zhǎng)圖芍躏。


原文網(wǎng)址:www.jb51.net/article/108839.htm

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末邪乍,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子对竣,更是在濱河造成了極大的恐慌庇楞,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件否纬,死亡現(xiàn)場(chǎng)離奇詭異吕晌,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)临燃,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén)睛驳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)烙心,“玉大人,你說(shuō)我怎么就攤上這事乏沸∫穑” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵蹬跃,是天一觀的道長(zhǎng)匙瘪。 經(jīng)常有香客問(wèn)我,道長(zhǎng)蝶缀,這世上最難降的妖魔是什么丹喻? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮翁都,結(jié)果婚禮上碍论,老公的妹妹穿的比我還像新娘。我一直安慰自己柄慰,他們只是感情好骑冗,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著先煎,像睡著了一般贼涩。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上薯蝎,一...
    開(kāi)封第一講書(shū)人閱讀 49,111評(píng)論 1 285
  • 那天遥倦,我揣著相機(jī)與錄音,去河邊找鬼占锯。 笑死袒哥,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的消略。 我是一名探鬼主播堡称,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼艺演!你這毒婦竟也來(lái)了却紧?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤胎撤,失蹤者是張志新(化名)和其女友劉穎晓殊,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體伤提,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡巫俺,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了肿男。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片介汹。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡却嗡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出嘹承,到底是詐尸還是另有隱情窗价,我是刑警寧澤,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布赶撰,位于F島的核電站舌镶,受9級(jí)特大地震影響柱彻,放射性物質(zhì)發(fā)生泄漏豪娜。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一哟楷、第九天 我趴在偏房一處隱蔽的房頂上張望瘤载。 院中可真熱鬧,春花似錦卖擅、人聲如沸鸣奔。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)挎狸。三九已至,卻和暖如春断楷,著一層夾襖步出監(jiān)牢的瞬間锨匆,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工冬筒, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留恐锣,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓舞痰,卻偏偏與公主長(zhǎng)得像土榴,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子响牛,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,519評(píng)論 25 707
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫(kù)玷禽、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,029評(píng)論 4 62
  • 歡迎來(lái)到標(biāo)簽時(shí)代 不知道從什么時(shí)候起坯台,出生年代和出生日期都成了一個(gè)人的標(biāo)簽,如00后瘫寝、90后蜒蕾、處女座等稠炬。作為一個(gè)基...
    無(wú)銀銀閱讀 661評(píng)論 3 5
  • 命里寄我一具腐骨 此身卻傍骨開(kāi)了夏蓮 朝起生花 暮落朽爛 開(kāi)敗里數(shù)落著無(wú)常 ——命數(shù)、命數(shù) 卻是紅顏枯骨 以為手里...
    木子慢閱讀 194評(píng)論 1 1
  • 八月,趕覺(jué)真是很忙碌的撤摸。主要是工作毅桃,每天感覺(jué)都做不完。睡覺(jué)了在回顧今天的工作計(jì)劃准夷,總還有那么一兩個(gè)沒(méi)能劃勾√钥飞。當(dāng)日...
    細(xì)思篤行閱讀 202評(píng)論 0 2