Android文档水印之PDF水印

随着移动终端办公化的普及,在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时,会显示水印,反之则不会显示。

上一篇:Java序列化心得(一):序列化设计和默认序列化格式的问题


下一篇:Android 跨应用间调用: URL Scheme