全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

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 页面

<html>
<head>
  <meta charset="utf-8"/>
</head>
<body>
<img src="mark.png" width="13px" height="20px"
   style="position:absolute;top: 0px;left: 12px;margin-bottom: 15px;"/>
<article id="content" style="margin: 25px;"></article>
<script type="text/javascript">
    function changeContent(content) {
      document.getElementById('content').innerHTML = content;
    }
</script>
</body>
</html>

这个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 = "<div style=\"color: gray;display: inline;\">" + content + "</div>";
    }
    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);

   /**
   * @param mode
   */
  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,只是为大家提供了一种更方便的功能,可以按自己喜欢的内容生成更有效的长图。

以上所述是小编给大家介绍的Android仿简书长按文章生成图片效果,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!


# android  # 长按文章生成图片  # Android实现合并生成分享图片功能  # android异步生成图片的示例代码  # Android栗子の图片验证码生成实例代码  # Android 实现图片生成卷角和圆角缩略图的方法  # Android实现用文字生成图片的示例代码  # Android 录制手机屏幕视频生成GIF图片实例详解  # Android自定义View绘制随机生成图片验证码  # Android生成带圆角的Bitmap图片  # android通过bitmap生成新图片关键性代码  # Android布局生成分享图片代码实例  # 两种  # 文档  # 就可以  # 这样一个  # 加载  # 这张  # 很简单  # 自定义  # 看一下  # 小编  # 象中  # 的是  # 是一个  # 法会  # 就会  # 是在  # 还没有  # 会有  # 我在  # 好了 


相关文章: 内网网站制作软件,内网的网站如何发布到外网?  建站之星如何防范黑客攻击与数据泄露?  如何选购建站域名与空间?自助平台全解析  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  高端建站三要素:定制模板、企业官网与响应式设计优化  网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?  香港服务器建站指南:外贸独立站搭建与跨境电商配置流程  定制建站流程解析:需求评估与SEO优化功能开发指南  七夕网站制作视频,七夕大促活动怎么报名?  济南企业网站制作公司,济南社保单位网上缴费步骤?  代购小票制作网站有哪些,购物小票的简要说明?  建站主机与虚拟主机有何区别?如何选择最优方案?  音乐网站服务器如何优化API响应速度?  PHP 500报错的快速解决方法  微信小程序 input输入框控件详解及实例(多种示例)  完全自定义免费建站平台:主题模板在线生成一站式服务  零基础网站服务器架设实战:轻量应用与域名解析配置指南  如何自定义建站之星模板颜色并下载新样式?  成都网站制作公司哪家好,四川省职工服务网是做什么用?  网站制作模板下载什么软件,ppt模板免费下载网站?  如何快速建站并高效导出源代码?  如何选择靠谱的建站公司加盟品牌?  如何高效完成独享虚拟主机建站?  建站主机数据库如何配置才能提升网站性能?  如何快速搭建虚拟主机网站?新手必看指南  建站主机选择指南:服务器配置与SEO优化实战技巧  品牌网站制作公司有哪些,买正品品牌一般去哪个网站买?  网站代码制作软件有哪些,如何生成自己网站的代码?  建站主机选购指南:核心配置优化与品牌推荐方案  深圳防火门网站制作公司,深圳中天明防火门怎么编码?  网站建设制作需要多少钱费用,自己做一个网站要多少钱,模板一般多少钱?  如何用5美元大硬盘VPS安全高效搭建个人网站?  如何通过VPS建站无需域名直接访问?  建站OpenVZ教程与优化策略:配置指南与性能提升  广德云建站网站建设方案与建站流程优化指南  南阳网站制作公司推荐,小学电子版试卷去哪里找资源好?  ,柠檬视频怎样兑换vip?  如何用wdcp快速搭建高效网站?  如何通过二级域名建站提升品牌影响力?  香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化  头像制作网站在线制作软件,dw网页背景图像怎么设置?  python的本地网站制作,如何创建本地站点?  如何在景安云服务器上绑定域名并配置虚拟主机?  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  宠物网站制作html代码,有没有专门介绍宠物如何养的网站啊?  制作网站的网址是什么,请问后缀为.com和.com.cn还有.cn的这三种网站是分别是什么类型的网站?  C++如何将C风格字符串(char*)转换为std::string?(代码示例)  清单制作人网站有哪些,近日“兴风作浪的姑奶奶”引起很多人的关注这是什么事情?  如何获取PHP WAP自助建站系统源码?  建站之星如何通过成品分离优化网站效率? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。