use crate::bindgen::{FPDF_DOCUMENT, FPDF_PAGE};
use crate::bindings::PdfiumLibraryBindings;
use crate::error::{PdfiumError, PdfiumInternalError};
use crate::page_object::PdfPageObject;
use crate::page_object_group::PdfPageGroupObject;
use crate::page_object_private::internal::PdfPageObjectPrivate;
use crate::page_objects_common::{
PdfPageObjectIndex, PdfPageObjectsCommon, PdfPageObjectsIterator,
};
use crate::page_objects_private::internal::PdfPageObjectsPrivate;
use std::os::raw::c_int;
pub struct PdfPageObjects<'a> {
page_handle: FPDF_PAGE,
document_handle: FPDF_DOCUMENT,
bindings: &'a dyn PdfiumLibraryBindings,
do_regenerate_page_content_after_each_change: bool,
}
impl<'a> PdfPageObjects<'a> {
#[inline]
pub(crate) fn from_pdfium(
page_handle: FPDF_PAGE,
document_handle: FPDF_DOCUMENT,
bindings: &'a dyn PdfiumLibraryBindings,
) -> Self {
Self {
page_handle,
document_handle,
bindings,
do_regenerate_page_content_after_each_change: false,
}
}
#[inline]
pub(crate) fn get_page_handle(&self) -> &FPDF_PAGE {
&self.page_handle
}
#[inline]
pub(crate) fn do_regenerate_page_content_after_each_change(
&mut self,
do_regenerate_page_content_after_each_change: bool,
) {
self.do_regenerate_page_content_after_each_change =
do_regenerate_page_content_after_each_change;
}
pub fn create_group<F>(&'a self, predicate: F) -> Result<PdfPageGroupObject<'a>, PdfiumError>
where
F: Fn(&PdfPageObject) -> bool,
{
let mut result = self.create_empty_group();
for mut object in self.iter().filter(predicate) {
result.push(&mut object)?;
}
Ok(result)
}
#[inline]
pub fn create_empty_group(&self) -> PdfPageGroupObject<'a> {
PdfPageGroupObject::from_pdfium(
self.page_handle,
self.document_handle,
self.bindings,
self.do_regenerate_page_content_after_each_change,
)
}
}
impl<'a> PdfPageObjectsPrivate<'a> for PdfPageObjects<'a> {
#[inline]
fn document_handle(&self) -> FPDF_DOCUMENT {
self.document_handle
}
#[inline]
fn bindings(&self) -> &'a dyn PdfiumLibraryBindings {
self.bindings
}
#[inline]
fn len_impl(&self) -> PdfPageObjectIndex {
self.bindings.FPDFPage_CountObjects(self.page_handle) as PdfPageObjectIndex
}
fn get_impl(&self, index: PdfPageObjectIndex) -> Result<PdfPageObject<'a>, PdfiumError> {
if index >= self.len() {
return Err(PdfiumError::PageObjectIndexOutOfBounds);
}
let object_handle = self
.bindings
.FPDFPage_GetObject(self.page_handle, index as c_int);
if object_handle.is_null() {
if let Some(error) = self.bindings.get_pdfium_last_error() {
Err(PdfiumError::PdfiumLibraryInternalError(error))
} else {
Err(PdfiumError::PdfiumLibraryInternalError(
PdfiumInternalError::Unknown,
))
}
} else {
Ok(PdfPageObject::from_pdfium(
object_handle,
Some(self.page_handle),
None,
self.bindings,
))
}
}
#[inline]
fn iter_impl(&'a self) -> PdfPageObjectsIterator<'a> {
PdfPageObjectsIterator::new(self)
}
fn add_object_impl(
&mut self,
mut object: PdfPageObject<'a>,
) -> Result<PdfPageObject<'a>, PdfiumError> {
object.add_object_to_page(self).and_then(|_| {
if self.do_regenerate_page_content_after_each_change {
if !self
.bindings
.is_true(self.bindings.FPDFPage_GenerateContent(self.page_handle))
{
if let Some(error) = self.bindings.get_pdfium_last_error() {
Err(PdfiumError::PdfiumLibraryInternalError(error))
} else {
Err(PdfiumError::PdfiumLibraryInternalError(
PdfiumInternalError::Unknown,
))
}
} else {
Ok(object)
}
} else {
Ok(object)
}
})
}
fn remove_object_impl(
&mut self,
mut object: PdfPageObject<'a>,
) -> Result<PdfPageObject<'a>, PdfiumError> {
object.remove_object_from_page().and_then(|_| {
if self.do_regenerate_page_content_after_each_change {
self.bindings.FPDFPage_GenerateContent(self.page_handle);
if let Some(error) = self.bindings.get_pdfium_last_error() {
Err(PdfiumError::PdfiumLibraryInternalError(error))
} else {
Ok(object)
}
} else {
Ok(object)
}
})
}
}