Enhancing PDF Functionality in Flutter Apps
Render PDFs in a Flutter app, enhancing user experience with zoom and drawing features for crisp text, clear images, and interactive annotations.
Introduction
The primary aim of this POC (Proof of Concept) is to demonstrate the capability of rendering PDF documents directly within the Flutter application itself, thereby eliminating the need for external applications. This feature enhances the app's autonomy and user experience.
Enhanced Zooming Functionality: An essential aspect of the POC involves implementing robust zooming functionality while preserving high-resolution display to ensure crisp and clear rendering of text and images within the PDF viewer.
Integrating Drawing Capabilities: A key objective of the POC is to incorporate drawing functionality directly into the PDF viewer. By enabling users to annotate or draw on the document, the application fosters greater interactivity and engagement, facilitating collaborative work and personalisation.
Findings
Despite the extensive research and trials, the POC encountered several challenges that hindered a fully successful implementation. Also, these findings are done mainly on Android but also follow for iOS. The findings are detailed below:
Platform Channels:
Platform channels in Flutter allow communication between Dart and the platform's native code (such as Android or iOS). This enables the use of native APIs/SDKs within a Flutter application. There are 3 types of platform channels : MethodChannel, EventChannel and BasicMessageChannel. For this POC, method channels were considered to integrate a native PDF viewer within the Flutter app.
Create a Dart file inside
lib
folder:
// native_pdf_viewer.dart
import 'package:flutter/services.dart';
class NativePdfViewer {
static const platform = MethodChannel('pdf_viewer_channel');
Future<void> openPDF(String filePath) async {
try {
final result = await platform.invokeMethod('openPdf', {'path': filePath});
print(result);
} on PlatformException catch (e) {
print("Failed to open PDF: '${e.message}'.");
}
}
}
Now, inside Android folder, follow this folder structure and follow as shown below:
// PdfViewerHandler.kt
package com.example.flutter_template
import androidx.core.content.FileProvider
import android.content.ActivityNotFoundException
import androidx.core.content.ContextCompat
import android.content.Context
import android.content.Intent
import android.net.Uri
import io.flutter.plugin.common.MethodCall
import io.flutter.plugin.common.MethodChannel
import io.flutter.plugin.common.MethodChannel.MethodCallHandler
import java.io.File
class PdfViewerHandler(private val context: Context) : MethodCallHandler {
override fun onMethodCall(call: MethodCall, result: MethodChannel.Result) {
when (call.method) {
"openPdf" -> {
val filePath = call.argument<String>("path")
openPdfFile(filePath)
result.success(null)
}
else -> result.notImplemented()
}
}
private fun openPdfFile(filePath: String?) {
if (filePath == null) {
// Handle error - file path not provided
return
}
val file = File(filePath)
val fileUri = FileProvider.getUriForFile(context, "${context.packageName}.provider", file)
val intent = Intent(Intent.ACTION_VIEW)
intent.setDataAndType(fileUri, "application/pdf")
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
try {
context.startActivity(intent)
} catch (e: ActivityNotFoundException) {
// Handle the case where no PDF viewer is found
}
}
}
// MainActivity.kt
package com.example.flutter_template
import io.flutter.embedding.android.FlutterActivity
import io.flutter.embedding.engine.FlutterEngine
import io.flutter.plugin.common.MethodChannel
import androidx.annotation.NonNull // For NonNull
import com.example.flutter_template.PdfViewerHandler
class MainActivity : FlutterActivity() {
override fun configureFlutterEngine(@NonNull flutterEngine: FlutterEngine) {
super.configureFlutterEngine(flutterEngine)
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, "pdf_viewer_channel")
.setMethodCallHandler(PdfViewerHandler(this))
}
}
Also, do not forget to check all the user read and write permissions inside AndroidManifest.xml
.
Result : PDF is opening in a different app rather than within the Flutter application, it suggests that the PDF is being opened using an intent (Android) or a URL scheme (iOS) that directs the system to use the default PDF viewer application instead of rendering it within your own app.
PlatformView : Rendering a native Android view inside a Flutter app is possible, but it requires embedding the native view into the Flutter widget tree. This can be achieved using the PlatformView widget in Flutter, which allows you to embed Android views directly into the Flutter UI.
A popular open-source PDF viewer, AndroidPdfViewer, was evaluated for this purpose, as we need to specify a package. In fact this is also being used underneath various famous pdf rendering packages in flutter like
flutter_pdfview
andpdfx
. However, a critical limitation is its lack of annotation capabilities, as highlighted in its issues section.Another PDF viewer considered was DrawOnPdf. This viewer supports drawing on PDFs but does not save the annotations. Saving annotations is complex, as it requires analyzing the correct pixels and page numbers where the user made changes. The lack of saving functionality makes it unsuitable for applications requiring persistent annotations.
Therefore, this approach was also discarded.
- pdfx: The pdfx package provided excellent zooming functionality and high-resolution rendering in an example application. But, since as mentioned above, pdfx uses AndroidPdfViewer underneath, it does not have capabilities to annotate.
import 'package:flutter/material.dart';
import 'package:pdfx/pdfx.dart';
class PdfViewerScreen extends StatefulWidget {
const PdfViewerScreen({
Key? key,
}) : super(key: key);
@override
State<PdfViewerScreen> createState() => _PdfViewerScreenState();
}
class _PdfViewerScreenState extends State<PdfViewerScreen> {
PdfControllerPinch? pdfPinchController;
@override
void initState() {
super.initState();
_initializePdfController();
}
Future<void> _initializePdfController() async {
pdfPinchController = PdfControllerPinch(
document: PdfDocument.openAsset('assets/sample.pdf'),
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text("Sample PDF"),
leading: IconButton(
icon: const Icon(Icons.arrow_back),
onPressed: () {
Navigator.of(context).pop();
},
),
),
body: PdfViewPinch(
controller: pdfPinchController!,
onDocumentError: print,
onDocumentLoaded: print,
),
);
}
}
Achievements
The POC successfully demonstrated the rendering of PDFs within a Flutter application using the pdfx
package, providing good zooming and high-resolution display. However, the implementation of annotation features remained unachieved.
Exploring Microsoft 365 PDF Viewer Integration
To overcome the limitations faced in the initial POC, the integration of the Microsoft 365 PDF Viewer via Android Method Channels was explored. This approach ensured that PDF files were opened using the Microsoft 365 PDF Viewer rather than any third-party application. The integration provided:
High-resolution rendering
Annotation capabilities
Robust zooming functionality
However, this method involves opening the PDF through a third-party app, which partially deviates from the goal of complete in-app autonomy.
// add these packages in pubspec.yaml
external_app_launcher: ^4.0.0
url_launcher: ^6.1.7
Inside lib
folder create a file that connects method channel with Dart:
// create a file native_pdf_viewer and add this
import 'package:flutter/services.dart';
class NativePdfViewer {
static const platform = MethodChannel('pdf_viewer_channel');
Future<void> openPDF(String filePath) async {
try {
final result = await platform.invokeMethod('openPdf', {'path': filePath});
print(result);
} on PlatformException catch (e) {
print("Failed to open PDF: '${e.message}'.");
}
}
}
To cater the UI, add below code inside main file:
// you can add this inside main file
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
title: Text(widget.title),
),
body: Center(
child: TextButton(
onPressed: () async {
bool isAppInstalled = await LaunchApp.isAppInstalled(
androidPackageName: 'com.microsoft.office.officehubrow',
);
if (isAppInstalled) {
NativePdfViewer().openPDF(filePath); // filepath will be your pdf filepath
} else {
final url = Uri.parse(
'https://play.google.com/store/apps/details?id=com.microsoft.office.officehubrow',
); // will redirect to playstore if app is not already installed
if (await canLaunchUrl(url)) {
await launchUrl(
url,
mode: LaunchMode.externalApplication,
);
} else {
print('Could not launch $url');
}
}
},
child: const Text("Open via Microsoft"),
),
),
);
}
Now, inside Android folder follow this structure as shown below:
// Inside MainActivity.kt
package com.example.pdf_viewer
import io.flutter.embedding.android.FlutterActivity
import io.flutter.embedding.engine.FlutterEngine
import io.flutter.plugin.common.MethodChannel
import androidx.annotation.NonNull // For NonNull
import com.example.flutter_template.PdfViewerHandler
class MainActivity : FlutterActivity() {
override fun configureFlutterEngine(@NonNull flutterEngine: FlutterEngine) {
super.configureFlutterEngine(flutterEngine)
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, "pdf_viewer_channel")
.setMethodCallHandler(PdfViewerHandler(this))
}
}
// Inside PdfViewerHandler.kt
package com.example.pdf_viewer
import androidx.core.content.FileProvider
import android.content.ActivityNotFoundException
import androidx.core.content.ContextCompat
import android.content.Context
import android.content.Intent
import android.net.Uri
import io.flutter.plugin.common.MethodCall
import io.flutter.plugin.common.MethodChannel
import io.flutter.plugin.common.MethodChannel.MethodCallHandler
import java.io.File
class PdfViewerHandler(private val context: Context) : MethodCallHandler {
override fun onMethodCall(call: MethodCall, result: MethodChannel.Result) {
when (call.method) {
"openPdf" -> {
val filePath = call.argument<String>("path")
if (filePath.isNullOrEmpty()) {
result.error("NO_FILE_PATH", "The file path was not provided.", null)
} else {
if (openPdfFile(filePath)) {
result.success(null)
} else {
result.error("NO_PDF_VIEWER_FOUND", "No PDF viewer found to open the file.", null)
}
}
}
else -> result.notImplemented()
}
}
private fun openPdfFile(filePath: String?): Boolean{
if (filePath.isNullOrEmpty()) return false
val file = File(filePath)
val fileUri: Uri = try {
FileProvider.getUriForFile(context, "${context.packageName}.provider", file)
} catch (e: IllegalArgumentException) {
// Handle error - file URI could not be generated
return false
}
val viewIntent = Intent(Intent.ACTION_VIEW).apply {
setDataAndType(fileUri, "application/pdf")
addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
// Try to use the specified package first
setPackage("com.microsoft.office.officehubrow")
}
return try {
context.startActivity(viewIntent)
true
} catch (e: ActivityNotFoundException) {
// Handle the case where no PDF viewer is found
false
}
}
}
Conclusion
The POC provided valuable insights into rendering and interacting with PDFs within a Flutter application. While the pdfx
package successfully addressed the zooming and resolution requirements, the annotation feature was not achievable within the Flutter app itself. Alternatively, leveraging the Microsoft 365 PDF Viewer offered a comprehensive solution but involved reliance on an external application.