随着移动终端办公化的普及,在Android手机中阅读文档已经是很普遍的事情了,但是有些文档可能是涉及到一些企业机密与敏感的内容,不希望随意扩散出去。为了防止重要的文档被截屏或者拍照而泄露,在文档中加上水印是一种常见的安全措施。本文将和大家分享一种不依赖第三文档浏览器(如WPS)的PDF文件水印生成方法,欢迎大家留言讨论。
水印内容
水印的内容应该包含两部分:
- 提示信息,提示用户该文件的解释权或者版权的归属,再或者是其内容不许外泄,违者必究等等;
- 当前阅读者身信息,这样,如果有泄露者截屏,就将带有自己身份信息的水印一起呈现在画面上,事后一旦发现有文件泄露,就可以通过水印判断泄露者是谁,从而追究其责任。
正因为水印要提前出当前用户的身份,所以每个终端上的水印是应该根据用户身份生成的,是各不相同,这就要求水印是在手机终端上生成的。
这就会带来另外一个问题,水印和文档要一起显示,如果解决方案依赖于某种第三方文档浏览器,就可能带来用户不安装该浏览器而水印生成失败的情况,所以文档的浏览应该是不依赖第三方应用的,是完全可控的。
PDF浏览
如果浏览PDF文档,这里可以向大家推荐一个库 Android PdfViewer,这个库提供一个可以解析PDF文档的View,帮我们浏览PDF文件. 其添加方式如下
compile 'com.github.barteksc:android-pdf-viewer:2.4.0'
具体使用方式会在下面的代码中介绍。
水印的生成
经过前面的铺垫,我们开始正式介绍水印的生成方式,方案其实很简单:
整个页面由两个View构成,第一个View用来显示PDF文档,第二个View显示水印并覆盖在第一个VIew之上。
提到一个控件覆盖再另一个控件之上,熟悉Android布局的朋友应该马上就会想到FrameLayout。没错,就是利用FrameLayout(帧布局),这种布局方式虽然没有任何的定位方式,其应用的场景也并不多,但是帧布局的大小由控件中最大的子控件决定,如果控件的大小一样大的话,那么同一时刻就只能看到最上面的那个组件,后续添加的控件会覆盖前一个。如果对帧布局还不是很熟悉的朋友,推荐查看FrameLayout(帧布局)
浏览带水印的PDF文件Ativity的布局activity_pdf_viewer.xml如下:
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout
android:id="@+id/fram"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<com.github.barteksc.pdfviewer.PDFView
android:id="@+id/pdf_view"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
<cn.test.android.WaterMarkView
android:id="@+id/water_mark_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@null"/>
</FrameLayout>
其中第一个控件是Android PdfViewer库中提供的PDFViewer空间,其负责显示PDF文档;第二个控件是我们自定义的WaterMarkView,在这个View中来显示水印。注意WaterMarkView的background设为@null,这样它就是透明的,用户就可以看到下面的PDFViewer。
WaterMarkView的代码如下:
public class WaterMarkView extends View {
private String name;
private String waterMark;
public WaterMarkView(Context context) {
super(context);
initWaterMark();
}
public WaterMarkView(Context context, AttributeSet attrs) {
super(context, attrs);
initWaterMark();
}
public WaterMarkView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
initWaterMark();
}
//生成水印文本
private void initWaterMark() {
String waterMark= "机密文件,拷贝必究" + "\r\n" + name;
}
public void setUserName(String userName){
this.name = userName;
}
//开始绘制水印
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
//获得页面尺寸
int width=getWidth();
int height=getHeight();
//TextPaint是paint的子类,该类可以很方便的进行文字的绘制
TextPaint textPaint = new TextPaint();
textPaint.setARGB(0x80, 0, 0, 0);//设置水印颜色
textPaint.setTextSize(20.0F);//设置水印字体大小
textPaint.setAntiAlias(true); // 抗锯齿
//参数意义分别为:文字内容、TextPaint对象、文本宽度、对齐方式、行距倍数、行距加数和是否包含内边距。
//这里比较重要的地方是设置文本宽度,当文本宽度比这个值大的时候就会自动换行。
StaticLayout layout = new StaticLayout(waterMark, textPaint,width,
Layout.Alignment.ALIGN_NORMAL, 1.0F, 0.0F, true);
float addWidth=36.0f,addHeight=24.0f;
//水印的位置
float[] x = new float[]{width / 4-addWidth , width * 3 / 4-addWidth, width / 4-addWidth, width* 3 / 4-addWidth};
float[] y = new float[]{height / 4-addHeight, height / 4-addHeight, height*3 / 4-addHeight, height * 3 / 4-addHeight};
//页面上绘制四个水印
for (int i = 0; i < 4; i++) {
canvas.save();
canvas.translate(x[i], y[i]);
canvas.rotate(15);
layout.draw(canvas);
canvas.restore();
}
}
}
WaterMarkView中绘制了四个文字水印,其水印内容包含当前用户名。下面我们在一个Activity中应用WaterMarkView和PDFView,其代码如下:
public class PdfViewerActivity extends Activity {
private String display;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_pdf_viewer);
PDFView pdfView=(PDFView)findViewById(R.id.pdf_view);
WaterMarkView markView = (WaterMarkView) findViewById(R.id.water_mark_view);
String fileFullName=getIntent().getStringExtra("fileFullName");
Int type = intent.getIntExtra("fileType",0);
markView.setUserName(intent.getStringExtra("userName"));
if(type!=0) //不显示水印
markView.setVisibility(View.GONE);
File file=new File(fileFullName);
try {
// 加载文件
pdfView.fromFile(file).load();
} catch (Exception ex) {
Toast.makeText(mContext,"文件不存在或已损坏",Toast.LENGTH_SHORT).show();
finish();
}
}
//静态方法启动该页面,type=0为显示水印,其他值为不显示水印
public static void openPdfFile(Context mContext,String fileFullName, int type, String userName) {
Intent intent = new Intent(mContext, PdfViewerActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.putExtra("fileFullName", fileFullName);
intent.putExtra("fileType",type)
intent.putExtra("userName",userName);
mContext.startActivity(intent);
}
}
PdfViewerActivity的布局文件就是一开始介绍的activity_pdf_viewer.xml,通过调用静态方法openPdfFile来传递参数并启动该页面。当type==0时,会显示水印,反之则不会显示。