前言
使用简书APP的同学都知道,简书有这样一个功能;文章页长按内容时底部会出现一个 生成图片分享 的按钮,点击之后就可以将当前的文章生成一张长图片;这张图片可以保存到本地或分享给好友,同时还可为图片设置成为白和黑两种风格,很有艺术范。个人一直很喜欢这个功能。
但是从某一个版本开始,这个功能开始有BUG了,生成的图片只有底部的固定标题,而没有文章内容,长图也变成了小短图。向简书意见反馈后,得到的回复是,使用点击分享按钮生成图片功能;分享菜单包含的生成长图功能的确是可以的。但是,还是很怀念之前长按生成图片的功能,所以作为一名程序猿;怀着好奇的心情,决定自己去实现这样一个功能.
效果预览
老规矩,首先看一下实现后的效果;虽然整体没有简书有范,个人感觉还是挺像的。
文章页实现
内容
文章页内容的实现,没有什么难点。布局总的来说很简单,包含户信息和文章信息的一个linearLayout,外加一个WebVIEw即可。数据是根据布局中所需的内容,封装了一个HTMLBean 对象,而这个对象的则是通过使用Jsoup 解析当前页面的HTML文档内容获得(这里使用Jsoup 方式获取简书网页内容,只是个人学习,没有其他用意)。具体实现可查看 源码
长按菜单实现
这里特意说一下,长按d出底部按钮的实现方式。一般情况下对于长按效果的实现,我们都会通过设置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" /><article ID="content" ></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的内容就是文章内容了。
GenScreenShotActivitymFakeWebVIEw = (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仿简书长按文章生成图片效果所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)