/light-novel-library_Wenku8_Android

轻小说文库(Wenku8)安卓版三方App,Material Design风格的轻小说阅读器(light novel reader)。现已完全开源!

Primary LanguageJavaGNU General Public License v2.0GPL-2.0

文档目录


写在前面

Material Design风格的轻小说文库App完全开源了!想实践Material Design的可以参考本项目的源代码,不清楚的、以及各种相关话题都可以发issue交流~

这个项目是我第一次写安卓App,首先看了一本入门书籍,是chinapub写书评赠送的《第一行代码 Android》,然后便开始着手写这个app了。从2014年10月份开始动手写,都是些零零散散的时间,写到12月底也就是0.5.2.0版本,那时基本上是啥都不会,写的代码也是烂的不行。后来内测用户群里面有个设计师提出来设计Material Design(后面简称MD)风格的app,于是就打算重新开发。也是拖啊拖,到7月底才完成新的app,如果满打满算的话,应该是历时一整个月,手写代码量2W行以上。

本app完全是出于兴趣以及作为一个项目经历,所以完全无偿开发、源代码也是完全开源!欢迎大家交流。之前迟迟不开源新的MD一方面是觉得写得太不容易了,各种谷歌查资料,还为红杏插件贡献了好几十 _(:3」∠)_ 另一方面是正式用git管理,Wenku8的站长也禁止将API公开,所以分来两个版本管理也很麻烦,还有一方面是因为代码非常乱,没有整理。

**注:**1.x版本源代码有部分是拷贝0520的,而0520的风格非常糟糕,比如AsyncTaskisLoading定义在类外这样。所以编码风格会有不统一,新写的代码我还是比较满意的,老的代码也是因为修改的需求不大,所以也就没有重构了。编码风格的建议也欢迎发issue~


为什么App里面说要尽快缓存感兴趣的小说?为什么现在这么紧急地开放源代码?

请参阅文档末尾的开发者的碎碎念


Repo目录说明

  1. design-source/

设计师提供的app制作效果图,大家可以参考学习~

  1. eclipse-android/

0.5.1.0030版本的源代码(eclipse+ADT版本),这份源代码阉割了API部分,所以编译后也无法正常获取数据。

  1. eclipse-android-old/

0.5.2.1典藏版的源代码(Android Studio版本),完整从私有库里拷贝出来的。release版本和正式版发布地址一样。

  1. graph-source/

老版的制图源文件,当时自己弄得,完全就是随性而无章法 =。=

  1. studio-android/

1.x正式版的所有源代码存放处,MD风格的App完整源码,可编译通过并正常运行!(Android Studio版本)


App程序结构简明

1.x 版本

新版的MD风格App,设计图纸在design-source/,工程文件在studio-android/

三方库的引用全部采用Gradle dependency,在混淆方面会直接忽略,加密强度低,不过开发起来各种方便~

主要视图

  • App启动界面

1.x-cover-sc

Model & Controller: activity/WelcomeActivity.java
View: layout/layout_welcome.xml
  • 侧栏菜单栏

1.x-custom-menu

Model & Controller: fragment/NavigationDrawerFragment.java
View: layout/layout_main_menu.xml
  • 最近更新列表

1.x-list-loading 1.x-list

Model & Controller: fragment/FavFragment.java
Adapter: adapter/NovelItemAdapter.java
View: layout/fragment_latest.xml
  • 搜索界面

1.x-search 1.x-searching 1.x-search-result

搜索界面
  Model & Controller: activity/SearchActivity.java
  Adapter: adapter/SearchHistoryAdapter.java
  View: layout/layout_search.xml
搜索结果界面
  Model & Controller: activity/SearchResultActivity.java
  Adapter: adapter/NovelItemAdapterUpdate.java
  View: layout/layout_search_result.xml
  • 排行榜列表

1.x-rank-list 1.x-rank-list2

Model & Controller: fragment/RKListFragment.java
Adapter: adapter/NovelItemAdapterUpdate.java
View: layout/fragment_rklist.xml + (layout/fragment_novel_item_list.xml)s
  • 本地书架(收藏)列表

1.x-fav

Model & Controller: fragment/FavFragment.java
Adapter: adapter/NovelItemAdapterUpdate.java
View: layout/fragment_novel_item_list.xml
  • 小说详细信息

1.x-info 1.x-info-menu 1.x-info-hotspot

Model & Controller: activity/NovolInfoActivity.java
View: layout/layout_novel_info.xml
  • 小说章节选择

1.x-chapter

Model & Controller: activity/NovolChapterActivity.java
View: layout/layout_novel_chapter.xml
  • 左右滑动阅读引擎

1.x-reader 1.x-reader-swipe 1.x-reader-menu 1.x-reader-dark 1.x-reader-vertical 1.x-image-view 1.x-reader-jump 1.x-reader-jump-show 1.x-reader-setting 1.x-font-custom 1.x-font-seek

Model & Controller: reader/activity/Wenku8ReaderActivityV1.java
View: layout/layout_reader_swipe_temp.xml + (layout/layout_reader_swipe_page.xml)s
  • 上下滑动阅读引擎

1.x-reader-horizental

Model & Controller: activity/VerticalReaderActivity.java
View: layout/layout_vertical_reader_temp.xml
  • 设置界面

1.x-config 1.x-switch-tc 1.x-cover-tc 1.x-config-tc

Model & Controller: fragment/ConfigFragment.java
View: layout/fragment_config.xml

源代码结构

studio-android/LightNovelLibrary/app/src/main
|   AndroidManifest.xml
|
+---assets/fonts
|       fzss-gbk.ttf 方正书宋GBK字体
|
+---java/org/mewx/wenku8
|       |   MyApp.java 自定义Application为了全局获取Context
|       |
|       +---activity
|       |       AboutActivity.java 关于界面
|       |       MainActivity.java 主界面
|       |       MenuBackgroundSelectorActivity.java 侧栏菜单背景选择界面
|       |       NovelChapterActivity.java 章节选择界面
|       |       NovelInfoActivity.java 小说信息界面
|       |       SearchActivity.java 搜索界面
|       |       SearchResultActivity.java 搜索结果框架界面
|       |       UserInfoActivity.java 用户信息界面
|       |       UserLoginActivity.java 用户登录界面
|       |       VerticalReaderActivity.java 上下滑动界面
|       |       ViewImageDetailActivity.java 查看大图界面
|       |       WelcomeActivity.java 启动界面
|       |
|       +---adapter
|       |       NovelItemAdapter.java 老版的小说项Adapter,每10项更新一次
|       |       NovelItemAdapterUpdate.java 更新的小说项Adapter,动态更新每一项
|       |       SearchHistoryAdapter.java 搜索历史Adapter
|       |
|       +---component
|       |       PagerSlidingTabStrip.java 排行榜自定义的标签类
|       |       ScrollViewNoFling.java 可控滑动阻尼的ScrollView
|       |
|       +---fragment
|       |       ConfigFragment.java 设置界面
|       |       FavFragment.java 本地书架框架
|       |       LatestFragment.java 最近更新
|       |       NavigationDrawerFragment.java 侧栏菜单
|       |       NovelItemListFragment.java 通用的小说列表界面(嵌入框架中)
|       |       RKListFragment.java 排行榜框架
|       |
|       +---global
|       |   |   GlobalConfig.java 全局设置(糟糕向),其中inAlphaBuild控制内测版/正式版
|       |   |
|       |   \---api
|       |           ChapterInfo.java 章节信息类
|       |           NovelItemInfo.java 小说信息类
|       |           NovelItemInfoUpdate.java 更新的小说信息类
|       |           NovelItemList.java 小说项列表类
|       |           NovelItemMeta.java 小说完整信息类
|       |           OldNovelContentParser.java 旧的小说内容解析器,解析成text和image
|       |           UserInfo.java 用户信息类
|       |           VolumeList.java 卷信息类
|       |           Wenku8API.java API类
|       |           Wenku8Error.java 错误信息类,后期的编码中定义的
|       |           Wenku8Parser.java 通用项目解析器
|       |
|       +---listener
|       |       MyItemClickListener.java RecyclerView的单击监听接口
|       |       MyItemLongClickListener.java RecyclerView的长按监听接口
|       |
|       +---reader 这边准备封装成的UniversalReaderActivity库的
|       |   +---activity
|       |   |       Wenku8ReaderActivityV1.java 左右滑动阅读界面
|       |   |
|       |   +---loader
|       |   |       WenkuReaderLoader.java 小说载入类的抽象类
|       |   |       WenkuReaderLoaderXML.java XML格式小说载入类
|       |   |
|       |   +---setting
|       |   |       WenkuReaderSettingV1.java 阅读设置类V1
|       |   |
|       |   +---slider 三方划屏库
|       |   |   |   SlidingAdapter.java
|       |   |   |   SlidingLayout.java
|       |   |   |
|       |   |   \---base
|       |   |           BaseSlider.java
|       |   |           OverlappedSlider.java
|       |   |           PageSlider.java
|       |   |           Slider.java
|       |   |
|       |   \---view
|       |           WenkuReaderPageBatteryView.java 电池View(未使用)
|       |           WenkuReaderPageView.java 单页小说View(效率低)
|       |
|       +---service
|       |       HeartbeatSessionKeeper.java 心跳包保持session类(未使用)
|       |
|       \---util
|               LightBase64.java 轻量级base64封装库
|               LightCache.java 轻量级文件操作库(容易OOM)
|               LightNetwork.java 轻量级网络通信库(容易OOM)
|               LightTool.java 轻量级工具集合类
|               LightUserSession.java 轻量级用户Session管理(包括账号密码加解密)
|               Logger.java 轻量级日志类(未使用)
|
\---res/
    |    ... 略
    \

0.5.2.1 典藏版

老版的App,启动和运行方面都比MD版本流畅,针对旧机型维护。Eclipse版本源码见eclipse-android/目录(非最新),Android Studio版本源码见eclipse-android-old/目录(典藏版最新),老版的图片资源都是用Fireworks做的,源文件见graph-source/目录。

预览图

0520-cover 0520-menu 0520-list 0520-info 0520-reader

源代码结构

这边引入了不少三方库,但是方法笨拙,主要采用复制、合并源代码的方式,所以文件目录会比较乱。但是加密强度高 _(:3」∠)_ 以后做商业软件还是建议这样操作,虽然麻烦,但是混淆之后烦的要命~

eclipse-android-old\LightNovelLibrary\src
+---com
|   +---davemorrissey
|   |   \---labs
|   |       \---subscaleview 分部加载图片的库,查看大图防止OOM
|   |               ImageViewState.java
|   |               ScaleImageView.java
|   |               SubsamplingScaleImageView.java
|   |
|   +---ecloud
|   |   \---pulltozoomview 下拉放大的视图,设置界面用的小苹果~
|   |           IPullToZoom.java
|   |           PullToZoomBase.java
|   |           PullToZoomListView.java
|   |           PullToZoomListViewEx.java
|   |           PullToZoomScrollView.java
|   |           PullToZoomScrollViewEx.java
|   |
|   +---facebook
|   |   \---rebound 物理、动画库,用途忘记了 =。= 大半年没动了,好像是侧栏菜单用的
|   |       |   AndroidSpringLooperFactory.java
|   |       |   BaseSpringSystem.java
|   |       |   OrigamiValueConverter.java
|   |       |   SimpleSpringListener.java
|   |       |   Spring.java
|   |       |   SpringConfig.java
|   |       |   SpringConfigRegistry.java
|   |       |   SpringListener.java
|   |       |   SpringLooper.java
|   |       |   SpringSystem.java
|   |       |   SpringSystemListener.java
|   |       |   SpringUtil.java
|   |       |   SteppingLooper.java
|   |       |   SynchronousLooper.java
|   |       |
|   |       \---ui
|   |               SpringConfiguratorView.java
|   |               Util.java
|   |
|   +---special
|   |   \---ResideMenu 侧边滑动菜单iOS风格的
|   |           ResideMenu.java
|   |           ResideMenuItem.java
|   |           TouchDisableView.java
|   |
|   \---zcw
|       \---togglebutton iOS风格的切换按钮
|               ToggleButton.java
|
+---me
|   \---imid
|       \---swipebacklayout 右滑返回layout
|           \---lib
|               |   SwipeBackLayout.java
|               |   Utils.java
|               |   ViewDragHelper.java
|               |
|               \---app
|                       SwipeBackActivity.java
|                       SwipeBackActivityBase.java
|                       SwipeBackActivityHelper.java
|                       SwipeBackPreferenceActivity.java
|
+---org
|   \---mewx
|       \---lightnovellibrary
|           +---activity
|           |       AboutActivity.java 关于界面
|           |       BookshelfFragment.java 本地书架
|           |       LibraryFragment.java 小说库(主界面入口)
|           |       MainActivity.java 主界面
|           |       NovelImageActivity.java 看大图界面
|           |       NovelInfoActivity.java 小说信息界面
|           |       NovelListActivity.java 小说列表界面
|           |       NovelReaderActivity.java 阅读器界面(上下滑动)
|           |       NovelSearchActivity.java 搜索界面
|           |       SettingFragment.java 设置
|           |       StartActivity.java 启动界面
|           |       Wenku8Fragment.java (未完成)
|           |
|           +---api
|           |       Wenku8Interface.java 你懂的
|           |
|           +---component
|           |   |   GlobalConfig.java 全局设置界面
|           |   |   MyApp.java 用于获取Context的全局Application
|           |   |   NovelContentParser.java 小说内容解析器
|           |   |   XMLParser.java XML解析器
|           |   |
|           |   \---adapter
|           |           EntryElement.java 书库分类项
|           |           EntryElementAdapter.java 书库分类项Adapter
|           |           NovelContentAdapter.java 小说内容Adapter
|           |           NovelElement.java 小说项
|           |           NovelElementAdapter.java 小说项Adapter
|           |           NovelElementSearch.java 小说搜索项
|           |           NovelElementSearchAdapter.java 小说搜索项Adapter
|           |           NovelIcon.java 小说封面
|           |           NovelIconAdapter.java 小说封面Adapter
|           |
|           \---util
|                   LightBase64.java 轻量级Base64库
|                   LightCache.java 轻量级文件操作库
|                   LightNetwork.java 轻量级网络通信库
|
\---uk
    \---co
        \---senab
            \---photoview 轻量级看图的View
                |   Compat.java
                |   DefaultOnDoubleTapListener.java
                |   IPhotoView.java
                |   PhotoView.java
                |   PhotoViewAttacher.java
                |
                +---gestures
                |       CupcakeGestureDetector.java
                |       EclairGestureDetector.java
                |       FroyoGestureDetector.java
                |       GestureDetector.java
                |       OnGestureListener.java
                |       VersionedGestureDetector.java
                |
                +---log
                |       Logger.java
                |       LoggerDefault.java
                |       LogManager.java
                |
                \---scrollerproxy
                        GingerScroller.java
                        IcsScroller.java
                        PreGingerScroller.java
                        ScrollerProxy.java

UMENG统计数据(2015/08/16残念)

1.x-statistic 0520-statistic

新版的app用户量10天增长到了1W用户量,全是托wenku8的福。照现在的增长速度速度估计用户量峰值是3W左右。


App存档文件结构

存档文件示例可以到release区下载查看~

总览

sdcard/wenku8
+---cache 完全由UIL接管的图片缓存文件夹
+---custom 用户自定义文件夹,可以放入自定义侧栏壁纸、自定义阅读字体、自定义阅读背景
|       .nomedia
|
+---imgs 小说封面
|       .nomedia
|       *.jpg
|
\---saves 存档文件夹
    |   avatar.jpg 登陆后的头像
    |   bookshelf_local.wk8 本地书架
    |   cert.wk8 简单加密的登陆用户名和密码
    |   read_saves.wk8 旧版的上下滑动阅读进度存档
    |   read_saves_v1.wk8 新版的左右滑动阅读进度存档
    |   search_history.wk8 搜索历史存档
    |   settings.wk8 设置存档
    |
    +---imgs 小说插图
    |       .nomedia
    |       *.jpg
    |
    +---intro 小说信息
    |       *.xml
    |
    \---novel 小说正文
            *.xml

存档流程

  • App开启后初始化UIL,接下来所有的图片都使用UIL载入,每次加载一个图片都会自动生成cache目录下的以CRC32为文件名缓存文件;
  • App启动即读取settings.wk8,每次设置被改变了也会立即保存设置;
  • 进入小说信息界面,会自动保存当前小说的封面于imgs目录
  • 小说的收藏功能会将aid写入saves/bookshelf_local.wk8,取消收藏会取消写入并删除本地的相关xml内容(图片不会删除);
  • 小说的下载功能分为4种:
    • 检查更新:仅更新当前小说在saves/intro中的文件,每个小说3个文件,更新后刷新Activity。
      • aid-intro.xml 小说信息摘要
      • aid-introfull.xml 小说完整介绍
      • aid-volume.xml 小说卷信息
    • 更新下载:先更新当前小说在saves/intro中的文件,然后按照aid-volume.xml中的信息,下载小说具体内容(存在saves/novel中),每次下载完毕后检查xml中是否含有图片,如果有图片则插入下载图片的任务**(这就是为什么下载的时候进度条会跳变了 2331)**;
    • 覆盖下载:与更新下载不同的是,这种方式是强制覆盖所有文件,而更新下载会跳过已下载的同名文件;
    • 分卷下载:更新下载的分卷版;
  • 小说在章节长按章节可以选择以下两种阅读引擎:
    • 左右翻页引擎V1:进入时读取saves/read_saves_v1.wk8,退出时保存进度;
    • 上下滑动引擎(旧):进入时读取saves/read_saves.wk8,退出时保存进度;
  • 调用搜索记录时会自动读取saves/search_history.wk8,每次搜索一次即保存最新搜索记录;
  • 用户登录后会生成saves/cert.wk8saves/avatar.jpg,每次启动app时自动登陆;若未联网,点击头像或有访问请求时会自动登录;若登陆失败,会删除这两个凭据文件;

技术细节

主要是存档的实现和兼容存档的规范。

saves/bookshelf_local.wk8

文件内容示例:

1098||1939||1924||1749||278

文件保存规则:

  1. 收藏只保存aid,aid之间用||分隔,读取时调用str.split("\\|\\|");
  2. 每个aid对应saves/intro目录下3个文件:aid-intro.xmlaid-introfull.xmlaid-volume.xml,如果缺少一个文件的话则会忽略该aid对应的书,可以通过检查更新(下拉书架、小说信息界面的检查更新)来修复该错误;

saves/read_saves.wk8

文件内容示例:

66761,,23799,,25575||66752,,134,,15090||63125,,1139,,3059||
63126,,904,,2824||63127,,11299,,157217

文件保存规则:

  1. 上下滑动阅读存档基本元素是cid,,position,,height
  • cid: chapter id,对应saves/novel里面的文件名;
  • position: 滚动的位置,当前屏幕顶部的位置(px);
  • height: ScrollView的总高度(px);
  1. 每个基本元素之间用||分隔,读取时调用str.split("\\|\\|");
  2. 每次记录阅读的最底部位置,往下翻再上翻只记录最底下位置;
  3. 一个误判区是100px,如果不小心点进去再退出,则不会记录,只有翻过100px才会记录进度;
  4. 每一章一个记录,只添加不删除 =。=

saves/read_saves_v1.wk8

文件内容示例:

1863:65378:65380:10:0||1922:67426:67427:0:0||1656:56193:56194:38:0||
1244:38825:38827:0:0||1163:35537:35538:0:0||1151:35126:35128:0:0||
1247:38920:38922:230:0||1759:60160:60161:159:0||1886:66259:66261:0:0||
1016:51951:51952:0:0||1749:59813:59814:0:0||1575:52853:52855:0:0||
278:10165:10166:36:0||1213:37499:37503:0:0||1928:67584:67585:0:0||
1446:47586:47588:90:0||1755:60043:60044:0:0||1519:50813:50814:0:0||
1932:67774:67782:255:55||1701:57929:57935:0:0

文件保存规则:

  1. 基本元素是aid:cid:vid:paraid:wordid
  • aid: 小说id
  • cid: 卷id
  • vid: 章id
  • paraid: 段id(从0开始)
  • wordid: 字在段中的id(从0开始)
  1. 每个基本元素之间用||分隔,读取时调用str.split("\\|\\|");
  2. 每一本书一条记录,卷末自动删除记录,相当于阅读完毕;
  3. 记录的原理和分页算法相关的,分页算法是通过指定一个段落(paraid)一个起始字符(wordid),然后动态分页;

saves/search_history.wk8

文件内容示例:

[田中][1到][音][刀剑][1日日日][进击的巨人][光还在][人类衰退][古][寒蝉]

文件保存规则:

  1. 每一项通过[]分开,UTF-8无BOM编码;
  2. 0520版本会在开头加一个数字:
  • 0: 搜索作者,例如:[0田中]
  • 1: 搜索小说名,例如:[1到]
  1. 默认保存10条,在GlobalConfig.java中记录;

saves/settings.wk8

文件内容示例:

reader_line_distance::::10||||
menu_bg_path::::/storage/emulated/0/tencent/QQ_Images/5699e514d0bb9779.png||||
reader_background_path::::0||||
version::::1||||
reader_paragraph_distance::::14||||
language::::SC||||
menu_bg_id::::0||||
reader_font_size::::18||||
reader_font_path::::0

键的代码片段:

public enum SettingItems {
    version, // (int) 1
    language,
    menu_bg_id, // (int) 1-5 by system, 0 for user
    menu_bg_path, // (String) for user custom
    reader_font_path, // (String) path to ttf, "0" means default
    reader_font_size, // (int) sp (8 - 32)
    reader_line_distance, // (int) dp (0 - 32)
    reader_paragraph_distance, // (int) dp (0 - 48)
    reader_paragraph_edge_distance, // (int) dp (0 - 16)
    reader_background_path, // (String) path to an image, day mode only, "0" means default
}

文件保存规则:

  1. 采用键值对的形式,代码中使用ContentValue存储;
  2. 存储文件时,键与值用::::分隔,键值对与键值对用||||分隔;
  3. 存键的时候使用的是enum.toString()方法获取,避免硬编码产生的低级错误;

saves/cert.wk8

文件内容示例:

Z0M5a0daRXBsZG5SMFZpaFhlPT0K
|b1FQOUVXYjFja1d1SlRkazVtV3BSWGI9Cg==

文件保存规则:用户名|密码,换行是系统函数自动产生的,不影响读取。

加密流程:
    1. 原文: str
    2. 一遍加密: base64(str)
    3. 大小写互转: switch(base64(str))
    4. 两遍加密: base64(switch(base64(str)))
    5. 除了等号外,前后字符交换: swap(base64(switch(base64(str))))
    6. 三次加密: base64(swap(base64(switch(base64(str)))))

解密流程:
    同理

App二次开发指引

本App目前决定暂停维护了……

原因一方面是开发者自己要准备年底的研究生考试;另一方面也是最主要的方面,鹅厂大规模购置轻小说版权,这边有一些暂时不能公开的原因,总之本三方app得告一段落了,具体可以阅读开发者的碎碎念

前几天我用HTTrack抓取了lknovel.cn的数据库,总共20G,太过于庞大,文件数20W,我的xp爬虫机已卡死。但是由于服务器在国内,速度还不错,用1~2天扒完。

后来我又尝试抓取wenku8.com,服务器在美国,这边速度太慢了,抓了一整天才1G,我估计wenku8的数据量大约有30G,实在hold不住。

所以尽管API开源了,但是因为内陆速度实在太慢,我都没有兴趣抓取了,所以劝大家去抓其他站:
linovel.com 可以用HTTrack设置总目录页面为入口,深度5,然后设定MAX 10000000,图片采用探索模式即可(一般的扒站工具抓不下来图片),服务器没有防护;
lknovel.cn  可以用HTTrack设置章节页面及小说信息界面为入口(excel生成url即可),深度2,设定MAX 10000000,服务器没有防护。
wenku8.com  速度太慢了,数据量还大,插图还在文末,没啥性价比,别抓了。

也就是说以后本app将没有数据源了,如果想继续做app的话那就只能通过抓取html提纯的方式,如果有开发者感兴趣,想通过抓取的方式继续本app(比如说:可以同时抓取lknovel、linovel、wenku8的数据展示给用户),可以阅读下面的部分:

如何用其他文库的文档兼容本app

目前wenku8的书目2000不到,也就是说aid是4位数;linovellknovel的aid也都是4位数,而且都没有wenku8的大,所以:

在aid方面,可以采取aid+10000000的方式,比如linovel的aid是+1000万,lknovel的aid是+2000万,这样本地书架的内容就错开了。

在存档方面:

aid-intro.xml 可以采取两种文件格式:
    最小文件体:
    <?xml version="1.0" encoding="utf-8"?>
    <metadata>
    <data name="Title" aid="1749"><![CDATA[残酷童话]]></data>
    <data name="Author" value="仓桥由美子"/>
    <data name="BookStatus" value="已完成"/>
    <data name="LastUpdate" value="2015-08-01"/>
    <data name="IntroPreview"><![CDATA[  现实残酷,童话幻灭
      在现实生活中,王子与...]]></data>
    </metadata>

    扩展文件体(可以自定义字段)但是已定义的字段有这些:
    <?xml version="1.0" encoding="utf-8"?>
    <metadata>
    <data name="Title" aid="5"><![CDATA[狼与香辛料(狼与辛香料)]]></data>
    <data name="Author" value="支仓冻砂"/>
    <data name="DayHitsCount" value="14"/>
    <data name="TotalHitsCount" value="394148"/>
    <data name="PushCount" value="22940"/>
    <data name="FavCount" value="6003"/>
    <data name="PressId" value="电击文库" sid="1"/>
    <data name="BookStatus" value="已完成"/>
    <data name="BookLength" value="2004567"/>
    <data name="LastUpdate" value="2012-02-08"/>
    <data name="LatestSection" cid="36097"><![CDATA[第十七卷 插图]]></data>
    </metadata>

aid-introfull.xml 的文件内容示例:
    旅行于各地贩卖并收购物品的商人克拉福·罗伦斯,拜访帕斯罗村并离开后,在自己的马车上发现了不知从哪里跑来的东西。
    拨开从帕斯罗村购买的小麦束后,竟然发现一位拥有兽耳与尾巴的美少女。
    少女自称为贤狼赫萝,是带给帕斯罗村长期丰收的少女。
    “虽然咱长久以来被尊为神,不过,咱就是咱,咱是赫萝。”见到少女的一只手变化成狼脚的罗伦斯,虽然一边怀疑赫萝的身份,但一边也答应让想回到出生遥远北方的少女一同旅行。

aid-volume.xml 文件内容示例:
    <?xml version="1.0" encoding="utf-8"?>
    <package>
    <volume vid="55995"><![CDATA[第一卷]]>
    <chapter cid="55996"><![CDATA[序章 Prologue]]></chapter>
    <chapter cid="55997"><![CDATA[第一话 A-part 死与不死]]></chapter>
    <chapter cid="55998"><![CDATA[第一话 B-part 死与不死]]></chapter>
    <chapter cid="55999"><![CDATA[第二话 A-part 杀人与异能]]></chapter>
    <chapter cid="56000"><![CDATA[第二话 B-part 杀人与异能]]></chapter>
    <chapter cid="56001"><![CDATA[第三话 A-part 命与心]]></chapter>
    <chapter cid="56002"><![CDATA[第三话 B-part 命与心]]></chapter>
    <chapter cid="56003"><![CDATA[插曲 Interlude]]></chapter>
    <chapter cid="56004"><![CDATA[终章 Epilogue]]></chapter>
    <chapter cid="56005"><![CDATA[后记]]></chapter>
    <chapter cid="56006"><![CDATA[插图]]></chapter>
    </volume>
    <volume vid="68031"><![CDATA[第二卷]]>
    <chapter cid="68032"><![CDATA[【序章】Prologue]]></chapter>
    <chapter cid="68033"><![CDATA[【第一话】First Story 别墅与杀人事件]]></chapter>
    <chapter cid="68034"><![CDATA[【插曲】~Interlude~ 梦与心的夹缝间]]></chapter>
    <chapter cid="68035"><![CDATA[【第二话】Second Story 内心与创伤]]></chapter>
    <chapter cid="68036"><![CDATA[【插曲】~Interlude~ 某人的黑暗]]></chapter>
    <chapter cid="68037"><![CDATA[【第三话】Third Story 紧张局势与众人目的]]></chapter>
    <chapter cid="68038"><![CDATA[【插曲】~Interlude~ 黑暗之中]]></chapter>
    <chapter cid="68039"><![CDATA[【第四话】Fourth Story 白天与夜晚]]></chapter>
    <chapter cid="68040"><![CDATA[【插曲】~Interlude~ 黑夜与白昼的夹缝间]]></chapter>
    <chapter cid="68041"><![CDATA[【终章】Epilogue]]></chapter>
    <chapter cid="68042"><![CDATA[后记]]></chapter>
    <chapter cid="68255"><![CDATA[插图]]></chapter>
    </volume>
    <volume vid="68043"><![CDATA[第三卷]]>
    <chapter cid="68044"><![CDATA[【序章】Prologue]]></chapter>
    <chapter cid="68045"><![CDATA[【第一话·前篇】First Story*First part 妹妹和传说]]></chapter>
    <chapter cid="68046"><![CDATA[【第一话·后篇】First Story*Latter part 妹妹和传说]]></chapter>
    <chapter cid="68047"><![CDATA[【插曲】~Interlude~ 她所身处的黑暗]]></chapter>
    <chapter cid="68048"><![CDATA[【第二话·前篇】Second Story*First part 真实与冒牌货]]></chapter>
    <chapter cid="68049"><![CDATA[【第二话·后篇】Second Story*Latter part 真实与冒牌货]]></chapter>
    <chapter cid="68050"><![CDATA[【插曲】~Interlude~ 她所身处的黑暗]]></chapter>
    <chapter cid="68051"><![CDATA[【终章】~Story of End~ 于是来到她的生日]]></chapter>
    <chapter cid="68052"><![CDATA[【章外篇】~Epilogue~]]></chapter>
    <chapter cid="68053"><![CDATA[后记]]></chapter>
    <chapter cid="68256"><![CDATA[插图]]></chapter>
    </volume>
    </package>

vid.xml 小说内容示例(目前版本支持图文混排,自动提纯多余换行和前导空格):
    一场死亡游戏
    即将揭开序幕
    SAO玩家·桐人,以完全攻略为目标,
    在游戏舞台「艾恩葛朗特」城堡里展开一连串严酷的冒险。
    途中与女剑士·亚丝娜的相遇,也为桐人带来命中注定的契机——
    川原砾
    出身于光之国度,居住在亚兹罗斯。人生就是独行剑士。
    虽然嘴里一直逞强说自己不需要一起组队的伙伴,但是最近因为许多任务的难易度提高而感到相当棘手。虽然一辈子都没办法单独去唱KTV,但希望至少能升级到独自去吃烧肉的等级。
    <!--image-->http://pic.wenku8.com/pictures/0/471/17513/3213.jpg<!--image-->
    <!--image-->http://pic.wenku8.com/pictures/0/471/17513/3214.jpg<!--image-->
    <!--image-->http://pic.wenku8.com/pictures/0/471/17513/3215.jpg<!--image-->
    <!--image-->http://pic.wenku8.com/pictures/0/471/17513/3216.jpg<!--image-->

只要将抓取的内容通过正则表达式或者可更新的lua脚本,就可以实现抓包了,那么文库主界面可以换成这样:

文库入口(Material Card):
    轻国文库
    轻之文库
    轻小说文库
    etc

点进去是搜索界面,然后搜索、下载、转换成可以识别的格式这样。目前我打听到的是国内文库还是会免费维持下去的!

如果有学生党or开发者有兴趣,可以和我联系~


用到的开源库

- jgilfelt / SystemBarTint (Apache License 2.0)
    用于设置Kitkat以上版本StatusBar和NavigationBar颜色透明度等。
- nostra13 / Android-Universal-Image-Loader (Apache License 2.0)
    著名的UIL,用于管理图片缓存和加载的库,非常方便,不会OOM。
- astuetz / PagerSlidingTabStrip (Apache License 2.0)
    -> branch: jpardogo / PagerSlidingTabStrip
    Material风格的标签及页面库,jpardogo的分支在自定制方面功能更强。
- jpardogo / GoogleProgressBar (Apache License 2.0)
    -> branch: MewX / google-progress-bar
    谷歌风格的加载动画,我的分支添加了Google Doodle-notifier的样式。
    原版的样式是圆角矩形,这边间距细节什么的模仿的不是很完美。
- Google / Volley (Apache License 2.0)
    Google发布的大规模并发加载库。原版不支持byte返回值,我这边稍加修改了。
- afollestad / material-dialogs (MIT License)
    完美的Material Dialog兼容库!
- futuresimple / android-floating-action-button (Apache License 2.0)
    可以展开的FAB,动画效果很自然,但是不支持ripple比较遗憾,而且阴影会截断。
- vinc3m1 / RoundedImageView (Apache License 2.0)
    圆形ImageView,显示头像用的。
- chrisbanes / SmoothProgressBar (BEER-WARE LICENSE)
    平滑进度条,这里用在小说信息、搜索结果页面的ActionBar下部。
- davemorrissey / subsampling-scale-image-view (Apache License 2.0)
    支持局部加载的ImageView,用于小说查看大图,用一般的ImageView容易OOM。
- martiansutdio / SlidingLayout (Unknown License)
    针对电子书的划屏库,非常省内存,只有3页,自由度不高,但是毕竟拿来主义 =。=
- AnderWeb / discreteSeekBar (Apache License 2.0)
    动画效果不错的SeekBar,用在小说阅读的设置和跳转页面中。

开发者的碎碎念

本来app是不打算开源的,比如说代码写得不好,开源的话还要维护两个版本之类的理由……

主要还是学习Material走了很多弯路,很多尝试也都在历史push里面能看到,谷歌这边推出系统的Design,而且官方的app也开发的很好了,但是开发者真正要用的除了support-design库,就只有三方库了。不过各方面还是没有Google Play Store这款app做得好啊。我各种舍不得代码呢,不希望app被复制这种心态=。= 然而事实证明是我想多了……

不过,最近又接到各种死亡预告,于是这个app也快要寿终正寝了 233 现在完全开源,包括APPKEY啊、API啥的都开放了~欢迎大家交流学习~

分享一下这次开发app的经验教训吧:

  • 依赖于网站的app,一定要确保网站不会左右app发展。

    目前的情况是网站战略变化,导致本app的接口服务会停止,取而代之的是另一个合作的app(非三方)。之前说大家抓紧缓存的意思是,趁目前还能用,抓紧下载,指不定以后要付费啥的呢,照鹅厂这动作!

    所以我觉得,自己做app最好顺便把服务网站给一套做掉,就像空之文库那样有一份服务器镜像,这样LK出问题不会殃及空文。

  • 开发app类型要慎重。

    这边博文也写了,版权敏感的软件市场是不轻易收的,要提供版权证明。所以这边一直上不了架,轻文因为有原创的成分在,所以可以上架。现在国内审查越来越严了哎……

  • 用户操作指引很重要。

    这个app里面隐藏了很多小功能,比如:用户找不到搜索功能,长按章节可以选择左右翻页或者上下滑动阅读,长按历史搜索可以删除某条记录。又比如:音量键可以翻页、单击也可以翻页,还可以切换章节etc。就包括看图用户也不会,这边看图把按钮放在ActionBar上是为了避免与翻页操作冲突,但是用户说找不到图 =。= 也比方说自定义字体、图片什么的说了不能在外置SD卡,但是用户还是不能按照要求的操作。

    所以进入界面的时候应该用户友好,弹出一个操作提示,所以这方面是在下输了!

  • 原有的用户黏着性需要很强大的优势才能打破。

    本来轻小说有个app叫空之文库,我这边app出来之后,用户量确实增加了不少,但是插图在文末这点确实是最大的软肋,因为轻小说有不少的用户是中学生,这部分用户说难听点就是比较叽歪,但是为了用户量,作为替代品目标的app,确实需要在功能上满足绝大多数用户的需求。所以轻文出来之后,这边用户量确实流失了不少。


LICENSE

GNU GENERAL PUBLIC LICENSE

                     Version 2, June 1991

Copyright (C) 1989, 1991 Free Software Foundation, Inc., <http://fsf.org/>
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Everyone is permitted to copy and distribute verbatim copies
of this license document, but changing it is not allowed.

......