mirror of
https://git.pwmt.org/pwmt/zathura.git
synced 2025-01-01 10:56:00 +01:00
246 lines
8.5 KiB
ReStructuredText
246 lines
8.5 KiB
ReStructuredText
|
Example - A minimalistic PDF plugin
|
||
|
===================================
|
||
|
|
||
|
In this section we are going to develop a simplified version of the
|
||
|
`zathura-pdf-poppler <../zathura-pdf-poppler>`_ plugin. For the sake of
|
||
|
simplicity we are not discussing the build process of the plugin because
|
||
|
we would recommend you to adapt our Makefiles from existing plugins. In
|
||
|
addition we avoid most of the error handling that should be implemented.
|
||
|
|
||
|
Prerequisites
|
||
|
~~~~~~~~~~~~~
|
||
|
|
||
|
In order to use the following described functions and macros you have to
|
||
|
include the *plugin-api.h* header file:
|
||
|
|
||
|
::
|
||
|
|
||
|
#include <zathura/plugin-api.h>
|
||
|
|
||
|
This automatically loads other header files for the
|
||
|
*zathura\_document\_t*, *zathura\_page\_t* as well as all the other
|
||
|
types that are necessary automatically.
|
||
|
|
||
|
Register the plugin
|
||
|
~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
As previously described each plugin has to register itself to zathura so
|
||
|
that it can be used properly. Therefore we have introduced a macro
|
||
|
called *ZATHURA\_PLUGIN\_REGISTER* which expects several parameters:
|
||
|
|
||
|
- Plugin name *The name of the plugin*
|
||
|
- Major version *The plugins major version*
|
||
|
- Minor version *The plugins minor version*
|
||
|
- Revision *The plugins revision*
|
||
|
- Open function *The open function*
|
||
|
- Mimetypes *A character array of supported mime types*
|
||
|
|
||
|
In our case we are going to register our plugin "my plugin" with its
|
||
|
version 1.0.1, the register function *register\_functions* and the list
|
||
|
of supported mimetypes.
|
||
|
|
||
|
::
|
||
|
|
||
|
ZATHURA_PLUGIN_REGISTER(
|
||
|
"plugin-tutorial",
|
||
|
0, 1, 0,
|
||
|
register_functions,
|
||
|
ZATHURA_PLUGIN_MIMETYPES({
|
||
|
"application/pdf"
|
||
|
})
|
||
|
)
|
||
|
|
||
|
This macro will automatically generate among others a function called
|
||
|
*plugin\_register* which is used to register the plugin to zathura when
|
||
|
it has been loaded.
|
||
|
|
||
|
Register the plugin functions
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
In our macro we have defined that the function *register\_functions* is
|
||
|
used to install our functions which will implement a certain
|
||
|
functionality in the struct:
|
||
|
|
||
|
::
|
||
|
|
||
|
void
|
||
|
register_functions(zathura_plugin_functions_t* functions)
|
||
|
{
|
||
|
functions->document_open = plugin_document_open;
|
||
|
functions->document_free = plugin_document_free;
|
||
|
functions->page_init = plugin_page_init;
|
||
|
functions->page_clear = plugin_page_clear;
|
||
|
functions->page_render_cairo = plugin_page_render_cairo;
|
||
|
}
|
||
|
|
||
|
We are now going to give a short overview about the used functions in
|
||
|
the above code snippet. For a complete documentation you should checkout
|
||
|
the documentation of `zathura\_document\_functions\_t <../../doxygen>`_.
|
||
|
A document instance consists out of a *zathura\_document\_t* document
|
||
|
object that contains information about the document itself and a defined
|
||
|
number of *zathura\_page\_t* page objects. There are several functions
|
||
|
defined for those two types and they have to be implemented by the
|
||
|
plugin. For our simple plugin which will only be capable of rendering a
|
||
|
page we will need one function that is capable of opening the PDF
|
||
|
document and setting up all necessary objects for further usage and one
|
||
|
function which will clean up all the allocated objects afterwards. In
|
||
|
addition we need two of those functions for page objects as well and one
|
||
|
function that will actually implement the rendering process.
|
||
|
|
||
|
Open and closing a document
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
The first thing we have to do when opening a document is to initialize
|
||
|
all necessary objects and values that we are going to need for the
|
||
|
future use of the plugin. Therefore we have to implement our
|
||
|
*pdf\_document\_open* function:
|
||
|
|
||
|
::
|
||
|
|
||
|
zathura_error_t
|
||
|
plugin_document_open(zathura_document_t* document)
|
||
|
{
|
||
|
/* get path and password */
|
||
|
const char* path = zathura_document_get_path(document);
|
||
|
const char* password = zathura_document_get_password(document);
|
||
|
|
||
|
/* create document data */
|
||
|
char* uri = g_filename_to_uri(path, NULL, NULL);
|
||
|
PopplerDocument* poppler_document = poppler_document_new_from_file(uri, password, NULL);
|
||
|
g_free(uri);
|
||
|
|
||
|
if (poppler_document == NULL) {
|
||
|
return ZATHURA_ERROR_UNKNOWN;
|
||
|
}
|
||
|
|
||
|
/* save poppler document for further usage */
|
||
|
zathura_document_set_data(document, poppler_document);
|
||
|
|
||
|
/* get number of pages */
|
||
|
unsigned int number_of_pages = poppler_document_get_n_pages(poppler_document);
|
||
|
zathura_document_set_number_of_pages(document, number_of_pages);
|
||
|
|
||
|
return ZATHURA_ERROR_OK;
|
||
|
}
|
||
|
|
||
|
To open the document we retrieve the *path* and the optional *password*
|
||
|
of the document to create an instance of *PopplerDocument* which
|
||
|
represents a document in the poppler library. If this fails for any
|
||
|
reason (e.g.: the path does not exist, the user provided the incorrect
|
||
|
password) we tell zathura that this function failed for an unknown
|
||
|
reason. If we are lucky we continue and save the created
|
||
|
*poppler\_document* object in the custom data field of the document so
|
||
|
that we can access it later on. After that we determine the number of
|
||
|
pages that the document contains so that zathura can initialize every
|
||
|
single page.
|
||
|
|
||
|
Since we have allocated the *poppler\_document* object we have to make
|
||
|
sure that its resources will be freed when it is no longer needed. This
|
||
|
happens in our *pdf\_document\_free* function:
|
||
|
|
||
|
::
|
||
|
|
||
|
zathura_error_t
|
||
|
plugin_document_free(zathura_document_t* document, PopplerDocument* poppler_document)
|
||
|
{
|
||
|
g_object_unref(poppler_document);
|
||
|
|
||
|
return ZATHURA_ERROR_OK;
|
||
|
}
|
||
|
|
||
|
Page initialization
|
||
|
~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
Each page has to be initialized so that zathura knows about its
|
||
|
dimension. In addition this stage is used to store additional data in
|
||
|
the page that will be used for further use with it. Therefore we are
|
||
|
implementing *pdf\_page\_init* which will save the width and the height
|
||
|
of the page in the given structure:
|
||
|
|
||
|
::
|
||
|
|
||
|
zathura_error_t
|
||
|
plugin_page_init(zathura_page_t* page)
|
||
|
{
|
||
|
unsigned int page_index = zathura_page_get_index(page);
|
||
|
zathura_document_t* document = zathura_page_get_document(page);
|
||
|
PopplerDocument* poppler_document = zathura_document_get_data(document);
|
||
|
|
||
|
/* create poppler page */
|
||
|
PopplerPage* poppler_page = poppler_document_get_page(poppler_document, page_index);
|
||
|
zathura_page_set_data(page, poppler_page);
|
||
|
|
||
|
/* get page dimensions */
|
||
|
double width, height;
|
||
|
poppler_page_get_size(poppler_page, &width, &height);
|
||
|
|
||
|
zathura_page_set_width(page, width);
|
||
|
zathura_page_set_height(page, height);
|
||
|
|
||
|
return ZATHURA_ERROR_OK;
|
||
|
}
|
||
|
|
||
|
And we have to make sure that all requested resources are freed in the
|
||
|
end:
|
||
|
|
||
|
::
|
||
|
|
||
|
zathura_error_t
|
||
|
plugin_page_clear(zathura_page_t* page, PopplerPage* poppler_page)
|
||
|
{
|
||
|
g_object_unref(poppler_page);
|
||
|
|
||
|
return ZATHURA_ERROR_OK;
|
||
|
}
|
||
|
|
||
|
Render a page
|
||
|
~~~~~~~~~~~~~
|
||
|
|
||
|
After we have setup the document and the page objects we are ready to
|
||
|
implement the render function which finally will be able to draw our
|
||
|
page on a widget so that it can be viewed with zathura. This function
|
||
|
has two additional parameters to the already known *zathura\_page\_t*
|
||
|
object: One of them is a *cairo\_t* object which will be used to render
|
||
|
the page, the other one is a flag called *printing* which determines if
|
||
|
the rendered page should be rendered for the print process of zathura.
|
||
|
For instance if this flag is set to true you should not render any
|
||
|
rectangles around links in the document because they are totally
|
||
|
worthless on paper:
|
||
|
|
||
|
::
|
||
|
|
||
|
zathura_error_t
|
||
|
pdf_page_render_cairo(zathura_page_t* page, cairo_t* cairo, bool printing)
|
||
|
{
|
||
|
if (printing == false) {
|
||
|
poppler_page_render(poppler_page, cairo);
|
||
|
} else {
|
||
|
poppler_page_render_for_printing(poppler_page, cairo);
|
||
|
}
|
||
|
|
||
|
return ZATHURA_ERROR_OK;
|
||
|
}
|
||
|
|
||
|
In this case the *pdf\_page\_render\_cairo* function is very simplistic
|
||
|
since all the work is done by the *poppler* library. In your case you
|
||
|
might have to do some magic here to draw the page to the cairo object.
|
||
|
Make sure to check out the source code of our plugins.
|
||
|
|
||
|
Installation of the plugin
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
As we suggested earlier the easiest way to build and install the plugin
|
||
|
is to duplicate the *Makefile* (as long with its *common.mk* and
|
||
|
*config.mk* files of one of our plugins. It already contains all
|
||
|
necessary targets for building, installing and debugging the plugin.
|
||
|
|
||
|
Otherwise you could build the above plugin with the following command:
|
||
|
|
||
|
::
|
||
|
|
||
|
$ gcc -std=c99 -shared -fPIC -pedantic -Wall `pkg-config --cflags --libs poppler-glib zathura` \
|
||
|
-o pdf.so pdf.c
|
||
|
|
||
|
After that you have to copy the *pdf.so* file into the directory where
|
||
|
zathura looks for plugins (this is by default: */usr/lib/zathura*).
|