仿各种APP将文章DOM转JSON并在APP中以列表显示(android、ios、php已开源)
背景
一直以来都想实现类似新闻客户端、鲜城等文章型app的正文显示,即在web editor下编辑后存为json,在app中解析json并显示正文。
网上搜过,没找到轮子。都是给的思路,然后告知是公司项目不好分享代码,所以干脆就自己做。
例子给的ui很粗,以实现功能为目的,你要有兴趣可以star等我更新。



输出的效果看起来是如上图所示。包括web的编辑器、ios、android。没做ui美化。
原理
web端
只是为了验证功能,所以信息包括标题、内容、其他数据都是模拟的,输出的json格式看起来是这样的:
[
{
"id": 2,
"title": "fdfd",
"text": "[{\"object\":\"text1\",\"type\":2},{\"object\":\"http:\\/\\/gitwiduu.u.qiniudn.com\\/lanwen_14637283563254.jpg\",\"type\":3}]",
"author": "作者名称",
"created_at": "2016-05-20 15:12:38",
"updated_at": "2016-05-20 15:12:38"
},
{
"id": 3,
"title": "adfadf",
"text": "[{\"object\":\"text1adsfdasf\",\"type\":2}]",
"author": "作者名称",
"created_at": "2016-05-20 15:22:49",
"updated_at": "2016-05-20 15:22:49"
},
{
"id": 4,
"title": "adfadf",
"text": "[{\"object\":\"text1\",\"type\":2}]",
"author": "作者名称",
"created_at": "2016-05-20 15:23:07",
"updated_at": "2016-05-20 15:23:07"
}
]
web端基于laravel4.2开发,采用的umeditor和七牛云服务器上传图片(这部分已剥离单独开源),。我仔细研究了一下,可能需要更进一步自定义umeditor,把图片parse规则做好。
规则一句话:umeditor每个自然段以<p>标签包围,遍历<p>标签,找出文字和图片存为json即可。
public function getDom($text) {
$dom = new DOMDocument();
$dom->loadHTML($text);
$node = $dom->documentElement;
$tempArray = false;
//遍历所有节点
$childs = $node->getElementsByTagName('p');
foreach ($childs as $child) {
//文字
if ($child->nodeValue) {
$tempArray[] = ['object' => $child->nodeValue, 'type' => 2];
}
//图片
$imgs = $child->getElementsByTagName('img');
if ($imgs) {
foreach ($imgs as $img) {
$tempArray[] = ['object' => $img->attributes->getNamedItem('src')->nodeValue, 'type' => 3];
}
}
}
return json_encode($tempArray);
}
android端
android端基于RecyclerView,主要考虑的地方是在adapter中需要根据type来输出不同的view,我用之前独白故事项目的思路(是基于以前看过一篇文章大神的思路),把不同type类型封装为“render”,然后根据数据的type来确定显示什么view。
package com.huijimuhe.lanwen.adapter.base; import android.annotation.TargetApi;
import android.support.v7.widget.RecyclerView;
import android.view.View;
import android.view.ViewGroup; import java.util.ArrayList;
import java.util.List; public abstract class AbstractRenderAdapter<T> extends RecyclerView.Adapter<AbstractViewHolder> { public static final int BTN_CLICK_ITEM = 0; public ArrayList<T> mDataset;
public onItemClickListener mOnItemClickListener;
protected View mHeaderView; @TargetApi(4)
public AbstractViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) {
return null;
} @Override
public int getItemCount() {
return mHeaderView == null ? mDataset.size() : mDataset.size() + 1;
} public List<T> getList() {
return this.mDataset;
} public int getRealPosition(int position) {
return mHeaderView == null ? position : position - 1;
} public T getItem(int position) {
return mDataset.get(getRealPosition(position));
} public void setOnItemClickListener(onItemClickListener l) {
mOnItemClickListener = l;
} public interface onItemClickListener {
void onItemClick(View view, int postion, int type);
} public void setHeaderView(View view) {
mHeaderView = view;
} public View getHeaderView() {
return mHeaderView;
}
}
上面的代码是基础baseadapter,加了通用的点击事件
package com.huijimuhe.lanwen.adapter; import android.annotation.TargetApi;
import android.view.ViewGroup; import com.huijimuhe.lanwen.adapter.base.AbstractRender;
import com.huijimuhe.lanwen.adapter.base.AbstractRenderAdapter;
import com.huijimuhe.lanwen.adapter.base.AbstractViewHolder;
import com.huijimuhe.lanwen.adapter.render.ImageSectionRender;
import com.huijimuhe.lanwen.adapter.render.TextSectionRender;
import com.huijimuhe.lanwen.model.SectionBean; import java.util.ArrayList; public class SectionAdapter extends AbstractRenderAdapter<SectionBean> {
public static final int RENDER_TYPE_TEXT = 0;
public static final int RENDER_TYPE_IMAGE = 1; public static final int BTN_CLICK_ITEM = 101;
public static final int BTN_CLICK_IMAGE = 102; public SectionAdapter(ArrayList<SectionBean> statues) {
this.mDataset = statues;
} @TargetApi(4)
public AbstractViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) { //header view 的判断
AbstractViewHolder holder = super.onCreateViewHolder(viewGroup, viewType);
if (holder != null) {
return holder;
} switch (viewType) {
case RENDER_TYPE_TEXT: {
TextSectionRender head = new TextSectionRender(viewGroup, this);
AbstractViewHolder headHolder=head.getReusableComponent();
headHolder.itemView.setTag(android.support.design.R.id.list_item,head);
return headHolder;
}
case RENDER_TYPE_IMAGE: {
ImageSectionRender head = new ImageSectionRender(viewGroup, this);
AbstractViewHolder headHolder=head.getReusableComponent();
headHolder.itemView.setTag(android.support.design.R.id.list_item,head);
return headHolder;
}
default:
return null;
}
} @TargetApi(4)
public void onBindViewHolder(AbstractViewHolder holder, int position) {
AbstractRender render = (AbstractRender) holder.itemView.getTag(android.support.design.R.id.list_item);
render.bindData(position);
} @Override
public int getItemViewType(int position) {
int type = getItem(position).getType();
switch (type) {
case 2:
return RENDER_TYPE_TEXT;
case 3:
return RENDER_TYPE_IMAGE;
default:
return 0;
}
}
}
这是文章的adapter,可以看到getItemViewType重写了。
package com.huijimuhe.lanwen.adapter.render; import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView; import com.huijimuhe.lanwen.R;
import com.huijimuhe.lanwen.adapter.SectionAdapter;
import com.huijimuhe.lanwen.adapter.base.AbstractRender;
import com.huijimuhe.lanwen.adapter.base.AbstractRenderAdapter;
import com.huijimuhe.lanwen.adapter.base.AbstractViewHolder;
import com.huijimuhe.lanwen.model.SectionBean; public class TextSectionRender extends AbstractRender { private ViewHolder mHolder;
private AbstractRenderAdapter mAdapter; public TextSectionRender(ViewGroup parent, AbstractRenderAdapter adapter) {
this.mAdapter =adapter;
View v = LayoutInflater.from(parent.getContext())
.inflate(R.layout.listitem_text,parent,false);
this.mHolder=new ViewHolder(v,adapter);
} @Override
public void bindData(int position) {
SectionBean model=(SectionBean) mAdapter.getItem(position);
mHolder.mTvContent.setText(model.getObject());
} @Override
public AbstractViewHolder getReusableComponent() {
return this.mHolder;
} public class ViewHolder extends AbstractViewHolder{
public TextView mTvContent; public ViewHolder(View v,final AbstractRenderAdapter adapter) {
super(v);
mTvContent = (TextView) v.findViewById(R.id.item_text); v.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
adapter.mOnItemClickListener.onItemClick(v, getLayoutPosition(), SectionAdapter.BTN_CLICK_ITEM);
}
});
}
}
}
这是文字段落的render。
ios端
其他功能如网络访问、json转换不再讨论,都是很常见的。在ios下实现其实简单,只需要根据type返回不同的cell即可,但需要注意的是高度,我图省事用的固定高度,这一块会在后续的更新中修正。
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
HJSectionModel* section=(HJSectionModel*)self.data[indexPath.row];
switch (section.type) {
case :{
HJTextTableViewCell* cell=[tableView dequeueReusableCellWithIdentifier:textIdentifier forIndexPath:indexPath];
cell.text.text=section.object;
return cell;
}
case :{
HJImageTableViewCell* cell=[tableView dequeueReusableCellWithIdentifier:imageIdentifier forIndexPath:indexPath];
[cell.image sd_setImageWithURL:section.object];
return cell;
}
default:
return nil;
}
}
项目地址
【WEB】https://github.com/huijimuhe/dom2json-web
【ANDROID】https://github.com/huijimuhe/dom2json-android
【IOS】https://github.com/huijimuhe/dom2json-ios
结论
这样做和webview相比有几个好处:
1.读得快
2.你可以加点自定义控件进去,而且排版不别扭
3.排版可以更好的定制
如果你用react.js之类的h5,请无视这篇文章。其实我也想用,哈哈哈...
做完之后总结起来,下一步要改进的地方:
1.不能像鲜城一样几个图片存入一个数组
2.可以做个像投票和图片轮播的控件,这样才能显示出优势
3.android的adapter是还可以继续改进的
4.ios的架构可以写的更易读
P.S
来App独立开发群533838427
github:https://github.com/huijimuhe
仿各种APP将文章DOM转JSON并在APP中以列表显示(android、ios、php已开源)的更多相关文章
- 三、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-配置项目并实现IM登录
项目文章索引 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入框扩展面板的 ...
- 七、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-聊天消息项的实现
会话好友列表的实现 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入框扩展 ...
- 八、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-聊天输入框扩展面板的实现
聊天输入框扩展面板的实现 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入 ...
- 五、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-聊天输入框的实现
会话好友列表的实现 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入框扩展 ...
- 一、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-项目引言
项目文章索引 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入框扩展面板的 ...
- 二、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-腾讯云后台配置TXIM
项目文章索引 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入框扩展面板的 ...
- 四、Uniapp+vue+腾讯IM+腾讯音视频开发仿微信的IM聊天APP,支持各类消息收发,音视频通话,附vue实现源码(已开源)-会话好友列表的实现
会话好友列表的实现 1.项目引言 2.腾讯云后台配置TXIM 3.配置项目并实现IM登录 4.会话好友列表的实现 5.聊天输入框的实现 6.聊天界面容器的实现 7.聊天消息项的实现 8.聊天输入框扩展 ...
- 使用 jQuery Mobile 与 HTML5 开发 Web App 系列文章目录
使用 jQuery Mobile 与 HTML5 开发 Web App 系列文章目录 时间:2012年9月20日 分类:JavaScript 标签:HTML5‚ jQuery Mobile‚ Web ...
- iOS App 架构文章推荐
iOS应用开发架构 iOS应用架构谈系列 阿里技术沙龙 2.2.1. Hybrid App 2.2.2. taobao 客户端架构 2.2.3. alipay 客户端架构 iOS APP 架构漫谈 ...
随机推荐
- python之装饰器
一.简单装饰器: #定义装饰器函数 def W1(main_func): def outer(): print("before") main_func() print(" ...
- 继续说一下openjson 以及 json path 的使用 (2)
在openjson 里面,其实是可以把数据类型array里面的值遍历出来的,举个栗子 ) = N' {"name":"test", "obj" ...
- 论Top与ROW_NUMBER读取第一页的效率问题
10.29 前一段时间研究关于分页的问题,由于数据库属于百万级的,考虑了关于优化方面的问题.其中一个考虑是:第一页展现的频率肯定是最高的,所以我想第一页就使用Top N来读取. 这个想法本身是没有错, ...
- iOS沙盒路径变化的说明详解
最近用沙盒存储文件的时候发现了一个奇怪的现象,由于业务需要,我会将保存的文件绝对路径保存以便下次读取. 于是发现一个找不到的现象,即上一次保存下的绝对路径,再第二次打开app去查找的时候,发现找不到. ...
- linux 分区 物理卷 逻辑卷
今天我们主要说说分区.格式化.SWAP.LVM.软件RAID的创建哈~ 格式化 查看当前分区:fdisk -l 这个命令我们以前是讲过的,我现在问下,ID那项是什么意思? 83 是代表EXT2和E ...
- [转]Membership三步曲之入门篇 - Membership基础示例
本文转自:http://www.cnblogs.com/jesse2013/p/membership.html Membership三步曲之入门篇 - Membership基础示例 Members ...
- .Net程序员之Python基础教程学习----字符串的使用 [Second Day]
在The FirstDay 里面学习了列表的元组的使用,今天开始学习字符串的使用.字符串的使用主要要掌握,字符串的格式化(C语言中我们应该都知道,Python和C语言差别不大),字符串的基本 ...
- NOIP2008传纸条[DP]
题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸运的是 ...
- 第9章 用内核对象进行线程同步(3)_信号量(semaphore)、互斥对象(mutex)
9.5 信号量内核对象(Semaphore) (1)信号量的组成 ①计数器:该内核对象被使用的次数 ②最大资源数量:标识信号量可以控制的最大资源数量(带符号的32位) ③当前资源数量:标识当前可用资源 ...
- 面试题:return和finally执行
Demo类: public class Demo { public int get() { int x=1; try { x++; return x; }finally{ ++x; } } } Tes ...