summaryrefslogtreecommitdiff
path: root/platform/winrt/mupdf_cpp/MainPage.xaml.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'platform/winrt/mupdf_cpp/MainPage.xaml.cpp')
-rw-r--r--platform/winrt/mupdf_cpp/MainPage.xaml.cpp628
1 files changed, 312 insertions, 316 deletions
diff --git a/platform/winrt/mupdf_cpp/MainPage.xaml.cpp b/platform/winrt/mupdf_cpp/MainPage.xaml.cpp
index 36becdaa..9ef47e76 100644
--- a/platform/winrt/mupdf_cpp/MainPage.xaml.cpp
+++ b/platform/winrt/mupdf_cpp/MainPage.xaml.cpp
@@ -92,11 +92,12 @@ MainPage::MainPage()
m_linkcolor="#40AC7225";
mu_doc = nullptr;
m_docPages = ref new Platform::Collections::Vector<DocumentPage^>();
- m_printpages = ref new Platform::Collections::Vector<DocumentPage^>();
m_thumbnails = ref new Platform::Collections::Vector<DocumentPage^>();
m_page_link_list = ref new Platform::Collections::Vector<IVector<RectList^>^>();
m_text_list = ref new Platform::Collections::Vector<RectList^>();
m_linkset = ref new Platform::Collections::Vector<int>();
+
+ SetUpDirectX();
RegisterForPrinting();
CleanUp();
RecordMainThread();
@@ -104,6 +105,44 @@ MainPage::MainPage()
_pageLoadedHandlerToken = Loaded += ref new RoutedEventHandler(this, &MainPage::Page_Loaded);
}
+/* You need a Direct3D device to create a Direct2D device. This gets stuff
+ set up for Direct2D printing support */
+void MainPage::SetUpDirectX()
+{
+ UINT creation_flags = D3D11_CREATE_DEVICE_BGRA_SUPPORT;
+ ComPtr<IDXGIDevice> dxgi_device;
+ D2D1_FACTORY_OPTIONS options;
+ ZeroMemory(&options, sizeof(D2D1_FACTORY_OPTIONS));
+ D3D_FEATURE_LEVEL feature_levels[] =
+ {
+ D3D_FEATURE_LEVEL_11_1,
+ D3D_FEATURE_LEVEL_11_0,
+ D3D_FEATURE_LEVEL_10_1,
+ D3D_FEATURE_LEVEL_10_0,
+ D3D_FEATURE_LEVEL_9_3,
+ D3D_FEATURE_LEVEL_9_2,
+ D3D_FEATURE_LEVEL_9_1
+ };
+ ComPtr<ID3D11Device> device;
+ ComPtr<ID3D11DeviceContext> context;
+
+#if defined(_DEBUG)
+ options.debugLevel = D2D1_DEBUG_LEVEL_INFORMATION;
+#endif
+
+ ThrowIfFailed(D3D11CreateDevice(nullptr, D3D_DRIVER_TYPE_HARDWARE, 0,
+ creation_flags, feature_levels, ARRAYSIZE(feature_levels),
+ D3D11_SDK_VERSION, &device, &m_featureLevel, &context));
+ ThrowIfFailed(device.As(&m_d3d_device));
+ ThrowIfFailed(context.As(&m_d3d_context));
+ ThrowIfFailed(m_d3d_device.As(&dxgi_device));
+ ThrowIfFailed(D2D1CreateFactory(D2D1_FACTORY_TYPE_MULTI_THREADED,
+ __uuidof(ID2D1Factory1), &options, &m_d2d_factory));
+ ThrowIfFailed(CoCreateInstance(CLSID_WICImagingFactory, nullptr,
+ CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&m_wic_factory)));
+ m_d2d_factory->CreateDevice(dxgi_device.Get(), &m_d2d_device);
+}
+
/* Used during launch of application from file */
void MainPage::Page_Loaded(Object^ sender, RoutedEventArgs^ e)
{
@@ -215,6 +254,17 @@ void MainPage::Picker(Platform::Object^ sender, Windows::UI::Xaml::RoutedEventAr
if (!EnsureUnsnapped())
return;
+ /* If we are actively rendering a document to the print thread then notify the
+ user that they will need to wait */
+ if (m_print_active)
+ {
+ int total_pages = GetPrintPageCount();
+ auto str1 = "Cannot open new file. Currently rendering page " +
+ m_curr_print_count + " of " + total_pages + " for print queue";
+ NotifyUser(str1, StatusMessage);
+ return;
+ }
+
FileOpenPicker^ openPicker = ref new FileOpenPicker();
openPicker->ViewMode = PickerViewMode::List;
openPicker->SuggestedStartLocation = PickerLocationId::DocumentsLibrary;
@@ -408,7 +458,7 @@ void MainPage::CreateBlank(int width, int height)
DataWriterStoreOperation^ result = dw->StoreAsync();
/* Block on the Async call */
- while(result->Status != AsyncStatus::Completed) {
+ while(result->Status != Windows::Foundation::AsyncStatus::Completed) {
}
/* And store in a the image brush */
bmp->SetSource(ras);
@@ -438,8 +488,6 @@ void MainPage::CleanUp()
m_docPages->Clear();
if (m_thumbnails != nullptr && m_thumbnails->Size > 0)
m_thumbnails->Clear();
- if (m_printpages != nullptr && m_printpages->Size > 0)
- m_printpages->Clear();
/* With the ref counting this should not leak */
if (m_page_link_list != nullptr && m_page_link_list->Size > 0)
m_page_link_list->Clear();
@@ -474,6 +522,8 @@ void MainPage::CleanUp()
m_rectlist_page = -1;
m_Progress = 0.0;
m_doczoom = 1.0;
+ m_print_active = false;
+ m_curr_print_count = 1;
this->xaml_PageSlider->Minimum = m_slider_min;
this->xaml_PageSlider->Maximum = m_slider_max;
@@ -676,7 +726,6 @@ void MainPage::InitialRender()
doc_page->LinkBox = nullptr;
m_docPages->Append(doc_page);
m_thumbnails->Append(doc_page);
- m_printpages->Append(doc_page);
/* Create empty lists for our links and specify that they have
not been computed for these pages */
Vector<RectList^>^ temp_link = ref new Vector<RectList^>();
@@ -1556,95 +1605,79 @@ bool MainPage::IsNotStandardView()
xaml_WebView->Visibility == Windows::UI::Xaml::Visibility::Visible);
}
-/* The following code is for print support. This is just a simple demonstration of
- printing with MuPDF in the Windows 8 environment */
+/* The following code is for print support. */
void MainPage::RegisterForPrinting()
{
- m_printdoc = ref new PrintDocument();
- m_printdoc_source = m_printdoc->DocumentSource;
- m_printdoc->Paginate +=
- ref new Windows::UI::Xaml::Printing::PaginateEventHandler(this, &MainPage::CreatePrintPreviewPages);
- m_printdoc->GetPreviewPage +=
- ref new Windows::UI::Xaml::Printing::GetPreviewPageEventHandler(this, &MainPage::GetPrintPreviewPages);
- m_printdoc->AddPages +=
- ref new Windows::UI::Xaml::Printing::AddPagesEventHandler(this, &MainPage::AddPrintPages);
-
- PrintManager^ printMan = PrintManager::GetForCurrentView();
- m_printTaskRequestedEventToken =
- printMan->PrintTaskRequested +=
- ref new TypedEventHandler<PrintManager^, PrintTaskRequestedEventArgs^>(this, &MainPage::PrintTaskRequested);
-}
-
-void MainPage::UnregisterForPrinting()
-{
- // Remove the handler for printing initialization.
- PrintManager^ printMan = PrintManager::GetForCurrentView();
- printMan->PrintTaskRequested -= m_printTaskRequestedEventToken;
-}
-
-void MainPage::PrintTaskRequested(PrintManager^ sender, PrintTaskRequestedEventArgs^ e)
-{
- auto printTaskRef = std::make_shared<PrintTask^>(nullptr);
- *printTaskRef = e->Request->CreatePrintTask("MuPDF Printing",
- ref new PrintTaskSourceRequestedHandler([this, printTaskRef](PrintTaskSourceRequestedArgs^ args)
- {
- PrintTask^ printTask = *printTaskRef;
- PrintTaskOptionDetails^ printDetailedOptions =
- PrintTaskOptionDetails::GetFromPrintTaskOptions(printTask->Options);
-
- // Some standard printer options
- printDetailedOptions->DisplayedOptions->Clear();
- printDetailedOptions->DisplayedOptions->Append(Windows::Graphics::Printing::StandardPrintTaskOptions::MediaSize);
- printDetailedOptions->DisplayedOptions->Append(Windows::Graphics::Printing::StandardPrintTaskOptions::Copies);
-
- // Our custom options
- PrintCustomItemListOptionDetails^ resolution =
- printDetailedOptions->CreateItemListOption("resolution", "Render Resolution");
- resolution->AddItem("sres72", "72 dpi");
- resolution->AddItem("sres150", "150 dpi");
- resolution->AddItem("sres300", "300 dpi");
- resolution->AddItem("sres600", "600 dpi");
- resolution->TrySetValue("sres600");
- m_printresolution = 600;
- printDetailedOptions->DisplayedOptions->Append("resolution");
-
- PrintCustomItemListOptionDetails^ scaling = printDetailedOptions->CreateItemListOption("scaling", "Scaling");
- scaling->AddItem("sScaleToFit", "Scale To Fit");
- scaling->AddItem("sCrop", "Crop");
- // Add the custom option to the option list.
- printDetailedOptions->DisplayedOptions->Append("scaling");
- scaling->TrySetValue("sScaleToFit");
- m_printcrop = false;
- printTask->Options->MediaSize = PrintMediaSize::NorthAmericaLetter;
-
- PrintCustomItemListOptionDetails^ pageFormat = printDetailedOptions->CreateItemListOption(L"PageRange", L"Page Range");
- pageFormat->AddItem(L"PrintAll", L"Print all");
- pageFormat->AddItem(L"PrintRange", L"Print Range");
- printDetailedOptions->DisplayedOptions->Append(L"PageRange");
- PrintCustomTextOptionDetails^ pageRangeEdit = printDetailedOptions->CreateTextOption(L"PageRangeEdit", L"Range");
-
- printDetailedOptions->OptionChanged +=
- ref new TypedEventHandler<PrintTaskOptionDetails^, PrintTaskOptionChangedEventArgs^>(this, &MainPage::PrintOptionsChanged);
-
- // Invoked when the print job is completed.
- printTask->Completed += ref new TypedEventHandler<PrintTask^, PrintTaskCompletedEventArgs^>(
- [=](PrintTask^ sender, PrintTaskCompletedEventArgs^ e)
- {
- auto callback = ref new Windows::UI::Core::DispatchedHandler(
- [=]()
- {
- ClearPrintCollection();
- m_printpagedesc = PrintPageDesc();
- if (e->Completion == Windows::Graphics::Printing::PrintTaskCompletion::Failed)
- {
- NotifyUser("Sorry, printing failed", StatusMessage);
- }
- });
- Dispatcher->RunAsync(Windows::UI::Core::CoreDispatcherPriority::Normal, callback);
- });
- // Set the document source.
- args->SetSource(PrintDocumentSource);
- }));
+ m_print_manager = Windows::Graphics::Printing::PrintManager::GetForCurrentView();
+ m_print_manager->PrintTaskRequested +=
+ ref new TypedEventHandler<PrintManager^, PrintTaskRequestedEventArgs^>(this, &MainPage::SetPrintTask);
+}
+
+void MainPage::SetPrintTask(PrintManager^ sender, PrintTaskRequestedEventArgs^ args)
+{
+ PrintTaskSourceRequestedHandler^ source_handler =
+ ref new PrintTaskSourceRequestedHandler([this](PrintTaskSourceRequestedArgs^ args)-> void{
+ Microsoft::WRL::ComPtr<PrintPages> document_source;
+ ThrowIfFailed(Microsoft::WRL::MakeAndInitialize<PrintPages>(&document_source, reinterpret_cast<IUnknown*>(this)));
+ IPrintDocumentSource^ objSource(reinterpret_cast<IPrintDocumentSource^>(document_source.Get()));
+ args->SetSource(objSource);
+ });
+
+ PrintTask^ print_task =
+ args->Request->CreatePrintTask(L"MuPDF WinRT Print", source_handler);
+
+ /* Call backs so that we know when we are all done with the printing */
+ print_task->Progressing +=
+ ref new TypedEventHandler<PrintTask^, PrintTaskProgressingEventArgs^>(this, &MainPage::PrintProgress);
+ print_task->Completed +=
+ ref new TypedEventHandler<PrintTask^, PrintTaskCompletedEventArgs^>(this, &MainPage::PrintCompleted);
+ m_print_active = true;
+ m_curr_print_count = 0;
+
+ PrintTaskOptionDetails^ printDetailedOptions =
+ PrintTaskOptionDetails::GetFromPrintTaskOptions(print_task->Options);
+
+ // Some standard printer options
+ printDetailedOptions->DisplayedOptions->Clear();
+ printDetailedOptions->DisplayedOptions->Append(Windows::Graphics::Printing::StandardPrintTaskOptions::MediaSize);
+ printDetailedOptions->DisplayedOptions->Append(Windows::Graphics::Printing::StandardPrintTaskOptions::Copies);
+
+ // Our custom options
+ PrintCustomItemListOptionDetails^ resolution =
+ printDetailedOptions->CreateItemListOption("resolution", "Render Resolution");
+ resolution->AddItem("sres96", "96dpi");
+ resolution->AddItem("sres150", "150 dpi");
+ resolution->AddItem("sres300", "300 dpi");
+ resolution->AddItem("sres600", "600 dpi");
+ resolution->TrySetValue("sres600");
+ m_printresolution = 600;
+ printDetailedOptions->DisplayedOptions->Append("resolution");
+
+ PrintCustomItemListOptionDetails^ location = printDetailedOptions->CreateItemListOption("location", "Location");
+ location->AddItem("sCenter", "Center");
+ location->AddItem("sTopleft", "Top Left");
+ // Add the custom option to the option list.
+ printDetailedOptions->DisplayedOptions->Append("location");
+ location->TrySetValue("sCenter");
+ m_centerprint = true;
+ print_task->Options->MediaSize = PrintMediaSize::NorthAmericaLetter;
+
+ PrintCustomItemListOptionDetails^ pageFormat = printDetailedOptions->CreateItemListOption(L"PageRange", L"Page Range");
+ pageFormat->AddItem(L"PrintAll", L"Print all");
+ pageFormat->AddItem(L"PrintRange", L"Print Range");
+ printDetailedOptions->DisplayedOptions->Append(L"PageRange");
+ PrintCustomTextOptionDetails^ pageRangeEdit = printDetailedOptions->CreateTextOption(L"PageRangeEdit", L"Range");
+
+ printDetailedOptions->OptionChanged +=
+ ref new TypedEventHandler<PrintTaskOptionDetails^, PrintTaskOptionChangedEventArgs^>(this, &MainPage::PrintOptionsChanged);
+}
+
+int MainPage::GetPrintPageCount()
+{
+ if (m_ppage_num_list.size() > 0)
+ return m_ppage_num_list.size();
+ else
+ return m_num_pages;
}
void MainPage::PrintOptionsChanged(PrintTaskOptionDetails^ sender, PrintTaskOptionChangedEventArgs^ args)
@@ -1661,9 +1694,9 @@ void MainPage::PrintOptionsChanged(PrintTaskOptionDetails^ sender, PrintTaskOpti
IPrintOptionDetails^ resolution = sender->Options->Lookup(optionId);
String^ resolutionValue = safe_cast<String^>(resolution->Value);
- if (resolutionValue == "sres72")
+ if (resolutionValue == "sres96")
{
- m_printresolution = 72;
+ m_printresolution = 96;
}
else if (resolutionValue == "sres150")
{
@@ -1680,18 +1713,18 @@ void MainPage::PrintOptionsChanged(PrintTaskOptionDetails^ sender, PrintTaskOpti
}
/* Need to update preview with a change of this one */
- if (optionId == "scaling")
+ if (optionId == "location")
{
IPrintOptionDetails^ scaling = sender->Options->Lookup(optionId);
String^ scaleValue = safe_cast<String^>(scaling->Value);
- if (scaleValue == "sScaleToFit")
+ if (scaleValue == "sCenter")
{
- m_printcrop = false;
+ m_centerprint = true;
}
- if (scaleValue == "sCrop")
+ if (scaleValue == "sTopleft")
{
- m_printcrop = true;
+ m_centerprint = false;
}
force_reset = true;
}
@@ -1821,241 +1854,204 @@ void MainPage::RemovePageRangeEdit(PrintTaskOptionDetails^ printTaskOptionDetail
}
}
-void MainPage::ClearPrintCollection()
+void MainPage::CreatePrintControl(_In_ IPrintDocumentPackageTarget* docPackageTarget,
+ _In_ D2D1_PRINT_CONTROL_PROPERTIES* printControlProperties)
{
- m_printlock.lock();
- for (int k = 0; k < m_num_pages; k++)
- {
- auto thumb_page = this->m_thumbnails->GetAt(k);
- this->m_printpages->SetAt(k, thumb_page);
- }
- m_printlock.unlock();
+ m_d2d_printcontrol = nullptr;
+ ThrowIfFailed(m_d2d_device->CreatePrintControl(m_wic_factory.Get(), docPackageTarget,
+ printControlProperties, &m_d2d_printcontrol));
}
-/* Just use the thumbnail images for now */
-void MainPage::CreatePrintPreviewPages(Object^ sender, PaginateEventArgs^ e)
+void MainPage::DrawPreviewSurface(float width, float height, float scale_in,
+ D2D1_RECT_F contentBox, uint32 page_num,
+ IPrintPreviewDxgiPackageTarget* previewTarget)
{
- InterlockedIncrement64(&m_requestCount);
- PrintPageDesc ppageDescription;
- PrintTaskOptionDetails^ printDetailedOptions =
- PrintTaskOptionDetails::GetFromPrintTaskOptions(e->PrintTaskOptions);
- PrintPageDescription DeviceDescription =
- e->PrintTaskOptions->GetPageDescription(0);
-
- /* This has the media dimension */
- ppageDescription.pagesize = DeviceDescription.PageSize;
-
- ppageDescription.resolution.Width = DeviceDescription.DpiX;
- ppageDescription.resolution.Height = DeviceDescription.DpiY;
-
- ppageDescription.margin.Width = (std::max)(DeviceDescription.ImageableRect.Left,
- DeviceDescription.ImageableRect.Right -
- DeviceDescription.PageSize.Width);
-
- ppageDescription.margin.Height = (std::max)(DeviceDescription.ImageableRect.Top,
- DeviceDescription.ImageableRect.Bottom -
- DeviceDescription.PageSize.Height);
-
- ppageDescription.printpagesize.Width = DeviceDescription.PageSize.Width -
- ppageDescription.margin.Width * 2;
- ppageDescription.printpagesize.Height = DeviceDescription.PageSize.Height -
- ppageDescription.margin.Height * 2;
-
- ClearPrintCollection();
- PrintDocument^ printDocument = safe_cast<PrintDocument^>(sender);
- m_printpagedesc = ppageDescription;
+ int dpi = 96;
+ int index_page_num = page_num - 1;
+ int ren_page_num = index_page_num;
if (m_ppage_num_list.size() > 0)
- {
- printDocument->SetPreviewPageCount(m_ppage_num_list.size(),
- PreviewPageCountType::Intermediate);
- }
- else
- {
- printDocument->SetPreviewPageCount(m_num_pages,
- PreviewPageCountType::Intermediate);
- }
-}
-
-/* Set the page with the new raster information */
-void MainPage::UpdatePreview(int page_num, InMemoryRandomAccessStream^ ras,
- Point ras_size, Page_Content_t content_type, double zoom_in)
-{
- assert(IsMainThread());
+ ren_page_num = m_ppage_num_list[page_num - 1] - 1;
+
+ /* This goes on in a background thread. Hence is non-blocking for UI */
+ assert(IsBackgroundThread());
+
+ /* Set up all the DirectX stuff */
+ CD3D11_TEXTURE2D_DESC textureDesc(DXGI_FORMAT_B8G8R8A8_UNORM,
+ static_cast<uint32>(ceil(width * dpi / 96)),
+ static_cast<uint32>(ceil(height * dpi / 96)),
+ 1, 1, D3D11_BIND_RENDER_TARGET | D3D11_BIND_SHADER_RESOURCE);
+ ComPtr<ID3D11Texture2D> texture;
+ ThrowIfFailed(m_d3d_device->CreateTexture2D(&textureDesc, nullptr, &texture));
+ ComPtr<IDXGISurface> dxgi_surface;
+ ThrowIfFailed(texture.As<IDXGISurface>(&dxgi_surface));
+
+ // Create a new D2D device context for rendering the preview surface. D2D
+ // device contexts are stateful, and hence a unique device context must be
+ // used on each thread.
+ ComPtr<ID2D1DeviceContext> d2d_context;
+ ThrowIfFailed(m_d2d_device->CreateDeviceContext(D2D1_DEVICE_CONTEXT_OPTIONS_NONE,
+ &d2d_context));
+ // Update DPI for preview surface as well.
+ d2d_context->SetDpi(96, 96);
+
+ D2D1_BITMAP_PROPERTIES1 bitmap_properties =
+ D2D1::BitmapProperties1(D2D1_BITMAP_OPTIONS_TARGET | D2D1_BITMAP_OPTIONS_CANNOT_DRAW,
+ D2D1::PixelFormat(DXGI_FORMAT_B8G8R8A8_UNORM, D2D1_ALPHA_MODE_PREMULTIPLIED));
+
+ // Create surface bitmap on which page content is drawn.
+ ComPtr<ID2D1Bitmap1> d2d_surfacebitmap;
+ ThrowIfFailed(d2d_context->CreateBitmapFromDxgiSurface(dxgi_surface.Get(),
+ &bitmap_properties, &d2d_surfacebitmap));
+ d2d_context->SetTarget(d2d_surfacebitmap.Get());
+
+ /* Figure out all the sizing */
+ spatial_info_t spatial_info;
+ spatial_info.scale_factor = 1.0;
+ spatial_info.size.X = width;
+ spatial_info.size.Y = height;
+ Point ras_size = ComputePageSize(spatial_info, ren_page_num);
+ ras_size.X = ceil(ras_size.X);
+ ras_size.Y = ceil(ras_size.Y);
+
+ Array<unsigned char>^ bmp_data;
+ int code = mu_doc->RenderPageBitmapSync(ren_page_num, ras_size.X, ras_size.Y,
+ true, &bmp_data);
+ D2D1_SIZE_U bit_map_rect;
+ bit_map_rect.width = ras_size.X;
+ bit_map_rect.height = ras_size.Y;
+
+ D2D1_BITMAP_PROPERTIES1 bitmap_properties2 =
+ D2D1::BitmapProperties1(D2D1_BITMAP_OPTIONS_NONE,
+ D2D1::PixelFormat(DXGI_FORMAT_B8G8R8A8_UNORM, D2D1_ALPHA_MODE_PREMULTIPLIED));
+
+ ID2D1Bitmap1 *bit_map;
+ ThrowIfFailed(d2d_context->CreateBitmap(bit_map_rect, &(bmp_data[0]),
+ ras_size.X * 4, &bitmap_properties2,
+ &bit_map));
+ D2D1_SIZE_F size = bit_map->GetSize();
+
+ /* Handle centering */
+ float y_offset = 0;
+ float x_offset = 0;
+ if (m_centerprint)
+ {
+ y_offset = (height - size.height) / 2.0;
+ x_offset = (width - size.width) / 2.0;
+ }
+
+ d2d_context->BeginDraw();
+ d2d_context->DrawBitmap(bit_map, D2D1::RectF(x_offset, y_offset,
+ size.width + x_offset, size.height + y_offset));
+ ThrowIfFailed(d2d_context->EndDraw());
+ ThrowIfFailed(previewTarget->DrawPage(page_num, dxgi_surface.Get(), dpi, dpi));
+}
+
+HRESULT MainPage::ClosePrintControl()
+{
+ return (m_d2d_printcontrol == nullptr) ? S_OK : m_d2d_printcontrol->Close();
+}
+
+void MainPage::PrintPage(uint32 page_num, D2D1_RECT_F image_area, D2D1_SIZE_F page_area,
+ float device_dpi, IStream* print_ticket)
+{
+ int dpi = m_printresolution;
+ int index_page_num = page_num - 1;
+ int ren_page_num = index_page_num;
- WriteableBitmap ^bmp = ref new WriteableBitmap(ras_size.X, ras_size.Y);
- bmp->SetSource(ras);
+ /* Windoze seems to hand me a bogus dpi */
+ device_dpi = 96;
- DocumentPage^ doc_page = ref new DocumentPage();
- doc_page->Image = bmp;
-
- doc_page->Height = ras_size.Y;
- doc_page->Width = ras_size.X;
- doc_page->Content = content_type;
- doc_page->Zoom = zoom_in;
- this->m_printpages->SetAt(page_num, doc_page);
+ if (m_ppage_num_list.size() > 0)
+ ren_page_num = m_ppage_num_list[page_num - 1] - 1;
+
+ /* This goes on in a background thread. Hence is non-blocking for UI */
+ assert(IsBackgroundThread());
+
+ /* Print command list set up */
+ ComPtr<ID2D1DeviceContext> d2d_context;
+ ThrowIfFailed(m_d2d_device->CreateDeviceContext(D2D1_DEVICE_CONTEXT_OPTIONS_NONE,
+ &d2d_context));
+ ComPtr<ID2D1CommandList> clist;
+ ThrowIfFailed(d2d_context->CreateCommandList(&clist));
+ d2d_context->SetTarget(clist.Get());
+
+ /* Figure out all the sizing. Width and height here are for device_dpi */
+ float width = image_area.right - image_area.left;
+ float height = image_area.bottom - image_area.top;
+
+ spatial_info_t spatial_info;
+ spatial_info.scale_factor = 1.0;
+ /* width and height are based upon device dpi (96) and MuPDF native
+ resolution is 72dpi */
+ spatial_info.size.X = (width /device_dpi) * (m_printresolution);
+ spatial_info.size.Y = (height /device_dpi) * (m_printresolution);
+ Point ras_size = ComputePageSize(spatial_info, ren_page_num);
+ ras_size.X = ceil(ras_size.X);
+ ras_size.Y = ceil(ras_size.Y);
+
+ Array<unsigned char>^ bmp_data;
+ int code = mu_doc->RenderPageBitmapSync(ren_page_num, ras_size.X, ras_size.Y,
+ true, &bmp_data);
+ D2D1_SIZE_U bit_map_rect;
+ bit_map_rect.width = ras_size.X;
+ bit_map_rect.height = ras_size.Y;
+
+ D2D1_BITMAP_PROPERTIES1 bitmap_properties2 =
+ D2D1::BitmapProperties1(D2D1_BITMAP_OPTIONS_NONE,
+ D2D1::PixelFormat(DXGI_FORMAT_B8G8R8A8_UNORM, D2D1_ALPHA_MODE_PREMULTIPLIED));
+
+ ID2D1Bitmap1 *bit_map;
+ ThrowIfFailed(d2d_context->CreateBitmap(bit_map_rect, &(bmp_data[0]),
+ ras_size.X * 4, &bitmap_properties2,
+ &bit_map));
+ D2D1_SIZE_F size = bit_map->GetSize();
+
+ /* Handle centering */
+ float y_offset = 0;
+ float x_offset = 0;
+ if (m_centerprint)
+ {
+ /* Offsets need to be provided in the device dpi */
+ y_offset = (page_area.height - (size.height * device_dpi / m_printresolution)) / 2.0;
+ x_offset = (page_area.width - (size.width * device_dpi / m_printresolution)) / 2.0;
+ }
+
+ float image_height = (bit_map_rect.height / m_printresolution) * device_dpi;
+ float image_width = (bit_map_rect.width / m_printresolution) * device_dpi;
+
+ d2d_context->BeginDraw();
+ d2d_context->DrawBitmap(bit_map, D2D1::RectF(x_offset, y_offset,
+ image_width + x_offset, image_height + y_offset));
+ ThrowIfFailed(d2d_context->EndDraw());
+ ThrowIfFailed(clist->Close());
+ ThrowIfFailed(m_d2d_printcontrol->AddPage(clist.Get(), page_area, print_ticket));
}
-void MainPage::CleanUpPreview(int page_num)
+void MainPage::RefreshPreview()
{
- for (int k = page_num - 1; k <= page_num + 1; k++)
- {
- if (k >= 0 && k < this->m_num_pages && k != page_num)
- {
- auto doc = this->m_printpages->GetAt(k);
- if (doc->Content == THUMBNAIL) return;
-
- if (this->m_thumbnails->Size > k)
- {
- auto thumb_page = this->m_thumbnails->GetAt(k);
- this->m_printpages->SetAt(k, thumb_page);
- }
- }
- }
+ PrintPages *p_struct = (PrintPages*) m_print_struct;
+ p_struct->ResetPreview();
}
-void MainPage::RefreshPreview()
+/* This reference is needed so that we can reset preview when changes occur on options */
+void MainPage::SetPrintTarget(void *print_struct)
{
- auto callback =
- ref new Windows::UI::Core::DispatchedHandler([this]()
- {
- m_printdoc->InvalidatePreview();
- });
- Dispatcher->RunAsync(Windows::UI::Core::CoreDispatcherPriority::Normal, callback);
+ m_print_struct = print_struct;
}
-void MainPage::GetPrintPreviewPages(Object^ sender, GetPreviewPageEventArgs^ e)
+/* These are called by the print thread and as such we cannot do UI updates
+ directly. However we can keep a monitor of what is going on so that if
+ someone tries to open a new document while we are printing some monster doc,
+ we can let them know that they need to wait. */
+void MainPage::PrintProgress(PrintTask^ sender, PrintTaskProgressingEventArgs^ args)
{
- PrintDocument^ printDocument = safe_cast<PrintDocument^>(sender);
-
- LONGLONG requestNumber = 0;
- InterlockedExchange64(&requestNumber, m_requestCount);
- int index_page_num = e->PageNumber;
- int ren_page_num = index_page_num;
-
- if (m_ppage_num_list.size() > 0)
- {
- ren_page_num = m_ppage_num_list[index_page_num - 1];
- }
- auto doc_curr = this->m_printpages->GetAt(ren_page_num - 1);
- if (doc_curr->Content == PRINT_PREVIEW)
- {
- auto bitmap = doc_curr->Image;
- Image^ image = ref new Image();
- image->Source = bitmap;
- image->HorizontalAlignment = Windows::UI::Xaml::HorizontalAlignment::Center;
- image->VerticalAlignment = Windows::UI::Xaml::VerticalAlignment::Center;
- image->Stretch = Stretch::UniformToFill;
- image->Width = bitmap->PixelWidth;
- image->Height = bitmap->PixelHeight;
- printDocument->SetPreviewPage(index_page_num, image);
- }
- else
- {
- /* Determine the scale to fit case or crop */
- spatial_info_t spatial_info = InitSpatial(1.0);
- int pagNum = ren_page_num - 1;
- Point ras_size = ComputePageSize(spatial_info, pagNum);
- float scale = 1.0;
-
- if (!(this->m_printcrop))
- {
- float scaleY = m_printpagedesc.printpagesize.Height / ras_size.Y;
- float scaleX = m_printpagedesc.printpagesize.Width / ras_size.X;
- scale = (std::min)(scaleY, scaleX);
- }
- auto render_task =
- create_task(mu_doc->RenderPageAsync(pagNum, ras_size.X, ras_size.Y, true));
- render_task.then([=] (InMemoryRandomAccessStream^ ras)
- {
- UpdatePreview(pagNum, ras, ras_size, PRINT_PREVIEW, scale);
- auto doc_curr = this->m_printpages->GetAt(pagNum);
- auto bitmap = doc_curr->Image;
- Image^ image = ref new Image();
- image->Source = bitmap;
- image->HorizontalAlignment = Windows::UI::Xaml::HorizontalAlignment::Center;
- image->VerticalAlignment = Windows::UI::Xaml::VerticalAlignment::Center;
- image->Stretch = Stretch::UniformToFill;
- image->Width = bitmap->PixelWidth * scale;
- image->Height = bitmap->PixelHeight * scale;
- printDocument->SetPreviewPage(index_page_num, image);
- }, task_continuation_context::use_current());
- }
- this->CleanUpPreview(ren_page_num - 1);
+ assert(IsBackgroundThread());
+ this->m_curr_print_count = args->DocumentPageCount;
}
-void MainPage::AddPrintPages(Object^ sender, AddPagesEventArgs^ e)
+void MainPage::PrintCompleted(PrintTask^ sender, PrintTaskCompletedEventArgs^ args)
{
- PrintDocument^ printDocument = safe_cast<PrintDocument^>(sender);
-
- /* We create a list of tasks to create the pages for printing. Once all the
- pages are created then we can go ahead and start adding them for printing */
- std::vector<concurrency::task<void>> createPageTasks;
- double res_scale = ((double) m_printresolution / 72.0);
- int page_count = m_num_pages;
-
- if (m_ppage_num_list.size() > 0)
- page_count = m_ppage_num_list.size();
-
- for(int i = 0; i < page_count; ++i)
- {
- m_printlock.lock();
-
- int page_num = i;
- if (m_ppage_num_list.size() > 0)
- page_num = m_ppage_num_list[i] - 1;
-
- auto doc_curr = this->m_printpages->GetAt(page_num);
- if (doc_curr->Content != FULL_RESOLUTION)
- {
- spatial_info_t spatial_info = InitSpatial(res_scale);
- Point ras_size = ComputePageSize(spatial_info, page_num);
- float scale = 1.0;
-
- if (!(this->m_printcrop))
- {
- spatial_info = InitSpatial(1.0);
- ras_size = ComputePageSize(spatial_info, page_num);
- float scaleY = m_printpagedesc.printpagesize.Height / ras_size.Y;
- float scaleX = m_printpagedesc.printpagesize.Width / ras_size.X;
- scale = (std::min)(scaleY, scaleX);
- spatial_info = InitSpatial(res_scale * scale);
- ras_size = ComputePageSize(spatial_info, page_num);
- }
- auto render_task =
- create_task(mu_doc->RenderPageAsync(page_num, ras_size.X, ras_size.Y, true));
- createPageTasks.push_back(render_task.then([=] (InMemoryRandomAccessStream^ ras)
- {
- UpdatePreview(page_num, ras, ras_size, FULL_RESOLUTION, 1.0);
- }, task_continuation_context::use_current()));
- }
- m_printlock.unlock();
- }
-
- /* When all the tasks have finished then go ahead and add them to the print
- document */
- concurrency::when_all(createPageTasks.begin(), createPageTasks.end()). then([=]
- {
- for (int i = 0; i < page_count; i++)
- {
- int page_num = i;
- if (m_ppage_num_list.size() > 0)
- page_num = m_ppage_num_list[i] - 1;
- auto doc_curr = this->m_printpages->GetAt(page_num);
- auto bitmap = doc_curr->Image;
- Image^ image = ref new Image();
- image->Source = bitmap;
- image->HorizontalAlignment = Windows::UI::Xaml::HorizontalAlignment::Center;
- image->VerticalAlignment = Windows::UI::Xaml::VerticalAlignment::Center;
- image->Stretch = Stretch::UniformToFill;
- image->Width = bitmap->PixelWidth / res_scale;
- image->Height = bitmap->PixelHeight / res_scale;
- printDocument->AddPage(image);
- }
- /* All pages provided. */
- printDocument->AddPagesComplete();
- /* Reset the current page description as soon as possible since the
- PrintTask.Completed event might fire later */
- m_printpagedesc = PrintPageDesc();
- });
+ assert(IsBackgroundThread());
+ m_print_active = false;
}