Android仿简书长按文章生成图片效果

前言

使用简书APP的同学都知道,简书有这样一个功能:文章页长按内容时底部会出现一个 生成图片分享 的按钮,点击之后就可以将当前的文章生成一张长图片,这张图片可以保存到本地或分享给好友,同时还可为图片设置成为白和黑两种风格,很有艺术范。个人一直很喜欢这个功能。

但是从某一个版本开始,这个功能开始有bug了,生成的图片只有底部的固定标题,而没有文章内容,长图也变成了小短图。向简书意见反馈后,得到的回复是,使用点击分享按钮生成图片功能;分享菜单包含的生成长图功能的确是可以的。但是,还是很怀念之前长按生成图片的功能,所以作为一名程序猿;怀着好奇的心情,决定自己去实现这样一个功能.

效果预览

老规矩,首先看一下实现后的效果;虽然整体没有简书有范,个人感觉还是挺像的。

文章页实现

内容

文章页内容的实现,没有什么难点。布局总的来说很简单,包含户信息和文章信息的一个LinearLayout,外加一个WebView即可。数据是根据布局中所需的内容,封装了一个HtmlBean 对象,而这个对象的则是通过使用Jsoup 解析当前页面的HTML文档内容获得(这里使用Jsoup 方式获取简书网页内容,只是个人学习,没有其他用意)。具体实现可查看 源码

长按菜单实现

这里特意说一下,长按弹出底部按钮的实现方式。一般情况下对于长按效果的实现,我们都会通过设置View的OnLongClickListene事件去实现相应的功能,但是对于这里的WebView可以如下实现:

mWebView.setOnCreateContextMenuListener(new View.OnCreateContextMenuListener() {

    @Override

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

        genImg.setVisibility(View.VISIBLE);

        T.showSToast(mContext, "再次点击文章可隐藏图片分享");

    }

});

// 点击隐藏底部按钮

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;

    }

});

这里通过监听WebView的ContextMenu 监听何时显示底部按钮;同时在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);

    }

});

点击底部的Button就会跳转到生成长图的界面,同时将之前获取到的HTMLBean对象传递过去。

长图效果实现

这里首先说一下实现思路(思路来源于 此 )。

  • 首先通过WebView加载一个本地的Html页面,这个页面包含一些固定,定义了一些标签。然后根据传递过来的mHtmlBean 对象中的信息,通过执行JavaScript动态的替换静态HTML页面中的内容;

  • 关于黑白两种风格的实现,同样是WebView执行Js,动态替换HTML中CSS 样式,修改WebView的背景色呈现出两种不同的UI 效果。

  • 通过WebView的capturePicture 和Canvas 可以生成出当前WebView的Bitmap对象,有了这个Bitmap就可以图片保存的功能了。

好了,下面就通过代码分别实现上述步骤。

Html 页面

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

    function changeContent(content) {

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

}

这个HTML页面的内容很简单,在整个文档左上角放置了一个小角标,就是简书APP生成长图时的那个mark.

同时定义了一个JavaScript 方法,功能也很简单,就是用传递的参数content替换article标签中的文档内容。

自定义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("'", "\\'") + "\")");

    }

}

这几个方法是生成长图最核心的方法。在loadData 方法中首先调用了assembleData,这个方法会根据mHtmlBean 这个对象中的数据拼接出一段 HTML 文档。在webView的loadUrl 方法中会从本地加载之前定义好的JianShu.html这个页面。然后在页面加载完成,即onProgressChanged 回调方法中newProgress 的值等于100时调用updateView方法;这个方法会根据当前设置的模式,设置WebView的背景,如果是夜间模式,则会对assembleData 中生成的文档外部在添加 一个灰色风格的div标签,将整个内容包在这个div标签中,最后WebView执行JS方法 changeContent,传递的参数就是之前我们拼接好的内容。这样整个WebView又会刷新一次,整个WebView的内容就是文章内容了。

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对象通过上一个页面(文章页)传递的mHtmlBean 对象就可以更新当前视图了,同时可以通过RadioButton实现页面风格的切换。

保存图片

距离我们最后的目标 生成长图片 ,前面的工作可以说只是完成了50%,因为到目前为止我们只不过是在WebView中把整个文章内容加载出来而已;长图还没有呢。因此,下面的工作就是通过WebView 生成长图。

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 很人性化,通过这个方法,我们就可以获得当前WebView视图 可见与不可见 部分的Bitmap了。

其实通过WebView生成图片并不是一件难事,难得是如何把我们这里的图片保存下来;因为我们这里生成的是长图,如下图所示,这张照片的高度达到了惊人的。因此这里就要需要之前在 Bitmap 初探 中提到的第一种压缩方法进行文件大小的压缩了。具体实现,就不再重复贴出代码了,有兴趣的同学可参考 Github源码

到这里,我们就完全实现了仿照简书长按生成图片的功能。那么回过头再来看,这样一个功能,为什么在我的手机上,简书APP的长按功能会有bug呢。


缺陷

文章详情页的WebView是系统自带的WebView,在加载带 代码的文章时,没有对代码类的内容做特殊的解析,因此无法对代码高亮显示。只是最为普通的文本进行了显示,因此生成的长图中代码也是普通文本。简书APP还是高大上呀,对代码的高亮显示正是棒棒哒!


后话

一个偶然的机会,在尝试简书长按生成图片的功能时发现,原来简书是通过WebView选择的区域生成第二页的内容;因此当我在文章页空白区域长按后,点击生成图片时必然是只有空白的,只有底部的一些固定标签。因此,这应该不算是一个bug,只是为大家提供了一种更方便的功能,可以按自己喜欢的内容生成更有效的长图。


原文网址:www.jb51.net/article/108839.htm

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,009评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,087评论 4 62
  • 欢迎来到标签时代 不知道从什么时候起,出生年代和出生日期都成了一个人的标签,如00后、90后、处女座等。作为一个基...
    无银银阅读 665评论 3 5
  • 命里寄我一具腐骨 此身却傍骨开了夏莲 朝起生花 暮落朽烂 开败里数落着无常 ——命数、命数 却是红颜枯骨 以为手里...
    木子慢阅读 198评论 1 1
  • 八月,赶觉真是很忙碌的。主要是工作,每天感觉都做不完。睡觉了在回顾今天的工作计划,总还有那么一两个没能划勾√。当日...
    细思笃行阅读 203评论 0 2