Android实现App内直接预览本地PDF文件

在App内实现直接预览pdf文件,而不是通过调用第三方软件,如WPS office等打开pdf。

主要思路:通过PhotoView将pdf读取为图片流进行展示。

一、首先,获取对本地文件读取的权限

在AndrooidManifest.xml中声明权限,以及页面中动态获取权限。

二、在build.gradle中引入需要的包

    implementation 'com.github.chrisbanes:PhotoView:2.0.0'

三、在项目库地址中添加

以及:

四、添加页面布局文件

添加一个页面布局文件:activity_pdf.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout  xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <android.support.v4.view.ViewPager
        android:id="@+id/vp_pdf"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
    </android.support.v4.view.ViewPager>

</LinearLayout >

添加一个适配器布局:item_pdf.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.github.chrisbanes.photoview.PhotoView
        android:id="@+id/iv_pdf"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    </com.github.chrisbanes.photoview.PhotoView>


</LinearLayout>

五、打开和关闭的实现代码

public class PdfActivity extends AppCompatActivity {

    LayoutInflater mInflater;
    PdfRenderer mRenderer;
    ViewPager vpPdf;
    ParcelFileDescriptor mDescriptor;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_pdf);

        init();
    }

    public void init() {
        mInflater = LayoutInflater.from(this);
        vpPdf =findViewById(R.id.vp_pdf);

        try {
            openRender();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @TargetApi(Build.VERSION_CODES.LOLLIPOP)
    private void openRender() throws IOException {
    
        File file = new File(DataHelper.GetHelpPdfPath());

        //初始化PdfRender
        mDescriptor = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_ONLY);
        if (mDescriptor != null) {
            mRenderer = new PdfRenderer(mDescriptor);
        }

        //初始化ViewPager的适配器并绑定
        MyAdapter adapter = new MyAdapter();
        vpPdf.setAdapter(adapter);
    }

    class MyAdapter extends PagerAdapter {

        @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
        @Override
        public int getCount() {
            return mRenderer.getPageCount();
        }

        @Override
        public boolean isViewFromObject(View view, Object object) {
            return view==object;
        }

        @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
        @Override
        public Object instantiateItem(ViewGroup container, int position) {
            View view = mInflater.inflate(R.layout.item_pdf, null);

            PhotoView pvPdf = view.findViewById(R.id.iv_pdf);
            pvPdf.setEnabled(true);

            if (getCount() <= position) {
                return view;
            }

            PdfRenderer.Page currentPage = mRenderer.openPage(position);
            Bitmap bitmap = Bitmap.createBitmap(1080, 1760, Bitmap.Config.ARGB_8888);
            currentPage.render(bitmap, null, null, PdfRenderer.Page.RENDER_MODE_FOR_DISPLAY);
            pvPdf.setImageBitmap(bitmap);
            //关闭当前Page对象
            currentPage.close();

            container.addView(view);
            return view;
        }

        @Override
        public void destroyItem(ViewGroup container, int position, Object object) {
            //销毁需要销毁的视图
            container.removeView((View) object);
        }
    }
//关闭pdf
    @Override
    protected void onDestroy() {
        super.onDestroy();
        try{
            closeRenderer();
        }
        catch (Exception ex){
            ex.printStackTrace();
        }
    }
    //关闭pdf
    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    private void closeRenderer() throws IOException {
        if (mRenderer != null){
            mRenderer.close();
        }
        if (mDescriptor != null){
            mDescriptor.close();
        }
    }

}

<think>嗯,用户问的是android.support.annotation.NonNull是否已经不推荐使用了,以及新的用法是什么。首先,我需要确认这个注解的历史和现状。记得Android支持库中有很多注解,比如NonNull和Nullable,用于帮助开发者检测空值问题。 然后,想到Android Jetpack的出现,Google逐渐将支持库迁移到AndroidX,这是一个重大的重构。所以可能android.support包下的类都被迁移到AndroidX了。具体到NonNull注解,应该是在AndroidX中有了新的包名。 接下来需要确认是否android.support.annotation.NonNull已经被废弃。查阅官方文档或者Android开发者网站,发现确实在迁移到AndroidX后,推荐使用AndroidX中的注解库。原来的支持库注解已经被标记为deprecated,所以开发者应该改用新的包路径。 新的包名应该是androidx.annotation.NonNull,属于Android Jetpack的一部分。此外,可能需要检查依赖是否正确引入,比如在build.gradle中添加对androidx.annotation:annotation的依赖。 另外,可能用户在使用旧项目时遇到了迁移的问题,或者听说注解有变化,需要更新代码。这时候需要建议用户迁移到AndroidX,并使用新的注解,同时处理可能的迁移步骤,比如使用Android Studio的Refactor > Migrate to AndroidX功能。 还要注意,如果用户没有迁移到AndroidX,可能需要继续使用旧的支持库,但长期来看,迁移是必要的,因为支持库不再维护。所以回答中需要强调迁移的重要性,并提供具体的替换示例和依赖配置。 最后,确保回答结构清晰,分点说明旧注解的状态、新注解的位置、迁移步骤和依赖配置,这样用户能一步步理解和操作。同时提醒可能的常见问题,比如包名更改后的导入问题,以及如何自动替换代码中的旧注解。</think>是的,从Android支持库迁移到AndroidX后,`android.support.annotation.NonNull` 已被标记为废弃(deprecated)。以下是逐步说明和替代方案: --- ### 1. **旧注解的状态** - ❌ `android.support.annotation.NonNull` 属于旧版支持库 (`android.support` 包) - 自Android Jetpack推出后,Google已将支持库迁移至AndroidX架构 - 官方建议:所有新项目都应使用AndroidX注解 --- ### 2. **新的替代方案** - ✅ 迁移到AndroidX后,应使用新的包路径: ```java androidx.annotation.NonNull ``` - 功能完全一致,用于标记方法参数、返回值或字段不可为null --- ### 3. **迁移步骤** #### 方式一:手动替换(适用于少量代码) - 将代码中所有: ```java import android.support.annotation.NonNull; ``` 改为: ```java import androidx.annotation.NonNull; ``` #### 方式二:自动迁移(推荐) 1.Android Studio中选择菜单:`Refactor > Migrate to AndroidX` 2. 工具会自动替换所有支持库注解到AndroidX路径 3. 检查`build.gradle`文件是否已配置: ```gradle android { useAndroidX = true enableJetifier = true // 自动转换第三方库 } ``` --- ### 4. **依赖配置** 确保在`build.gradle`中包含AndroidX注解库: ```gradle dependencies { implementation 'androidx.annotation:annotation:1.8.0' // 使用最新版本 } ``` --- ### 5. **常见问题** - **Q**: 如果未迁移到AndroidX会怎样? - **A**: 可继续使用旧支持库,但将无法获得后续更新和安全修复 - **Q**: 第三方库仍用旧注解怎么办? - **A**: 启用`Jetifier`(见上述Gradle配置),它会自动转换依赖库的代码 --- ### 总结 迁移到 `androidx.annotation.NonNull` 是官方推荐的实践,建议尽快通过Android Studio的迁移工具完成升级,以确保项目兼容性和可维护性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值