diff --git a/src/plugins/dashboard/public/application/embeddable/viewport/_print_viewport.scss b/src/plugins/dashboard/public/application/embeddable/viewport/_print_viewport.scss
index a451178cc46b0..91a6b166700bc 100644
--- a/src/plugins/dashboard/public/application/embeddable/viewport/_print_viewport.scss
+++ b/src/plugins/dashboard/public/application/embeddable/viewport/_print_viewport.scss
@@ -1,9 +1,75 @@
-.printViewport {
- &__vis {
- height: 600px; // These values might need to be passed in as dimensions for the report. I.e., print should use layout dimensions.
- width: 975px;
+@import './shared_poc_print'; // Bring in the shared print styling
- // Some vertical space between vis, but center horizontally
- margin: 10px auto;
+/*
+Dashboard styling
+
+It is up to dashboard to decide how the visualizations should look on an A4 page. The assumption
+is that we want to preserve current behaviour of 2 visualizations per page.
+*/
+$visualisationsPerPage: 2;
+$visPadding: 4mm;
+
+/*
+We set the same visual padding on the browser and print versions of the UI so that
+we don't hit a race condition where padding is being updated while the print image
+is being formed. This can result in parts of the vis being cut out.
+*/
+@mixin visualizationPadding {
+ padding-left: $visPadding;
+ padding-right: $visPadding;
+
+ &:nth-child(#{$visualisationsPerPage}n - #{$visualisationsPerPage - 1}) {
+ padding-top: $visPadding;
+ }
+
+ &:nth-child(#{$visualisationsPerPage}n) {
+ page-break-after: always;
+ padding-top: $visPadding;
+ padding-bottom: $visPadding;
+ }
+}
+
+@include globalSharedRules();
+
+@media screen, projection {
+ .printViewport {
+ &__vis {
+ @include visualizationPadding();
+
+ & .embPanel__header button {
+ display: none;
+ }
+
+ margin: $euiSizeL auto;
+ height: calc(#{$a4PageContentHeight} / #{$visualisationsPerPage});
+ width: $a4PageContentWidth;
+ padding: $visPadding;
+ }
+ }
+}
+
+@media print {
+
+ .printViewport {
+ &__vis {
+ @include visualizationPadding();
+
+ height: calc(#{$a4PageContentHeight} / #{$visualisationsPerPage});
+ width: $a4PageContentWidth;
+
+ & .euiPanel {
+ box-shadow: none !important;
+ }
+
+ & .embPanel__header button {
+ display: none;
+ }
+
+ page-break-inside: avoid;
+
+ & * {
+ overflow: hidden !important;
+ }
+ }
}
}
diff --git a/src/plugins/dashboard/public/application/embeddable/viewport/_shared_poc_print.scss b/src/plugins/dashboard/public/application/embeddable/viewport/_shared_poc_print.scss
new file mode 100644
index 0000000000000..d4fae6dbc00a6
--- /dev/null
+++ b/src/plugins/dashboard/public/application/embeddable/viewport/_shared_poc_print.scss
@@ -0,0 +1,107 @@
+
+/*
+Global print styling
+
+This styling should be usable by an plugin/app that wants to provide print
+functionality. The styling should be extremely unopinionated.
+
+Observations:
+
+1. It is possible to request a print before all ES searches have populated
+ visualizations, this results in a borked print preview with half-rendered
+ visualisations. The user will have to close the print dialog and re-open
+ it. Not sure how to "fix" this UX.
+
+2. We currently do not control the user-agent's header and footer content
+ (including the style of fonts)
+
+3. I could not see a way to get page numbering other than using the
+ browser-provided footers :'(
+
+4. Page box model is quite different from what we have in browsers - page
+ margins define where the "no-mans-land" exists for actual content. Moving
+ content into this space by, for example setting negative margins resulted
+ in slightly unpredictable behaviour because the browser wants to either
+ move this content to another page or it just looks broken/split across two
+ pages
+
+5. page-break-* is your friend!
+*/
+$a4PageHeight: 297mm;
+$a4PageWidth: 210mm;
+$a4PageMargin: 0;
+$a4PagePadding: 0;
+$a4PageHeaderHeight: 5mm;
+$a4PageFooterHeight: 10mm;
+$a4PageMarginBottom: 9mm;
+
+$a4PageContentHeight: $a4PageHeight - $a4PageFooterHeight - $a4PageHeaderHeight - $a4PageMarginBottom;
+$a4PageContentWidth: $a4PageWidth;
+
+// Currently we cannot control or style the content the browser places in
+// margins, this might change in the future:
+// See https://drafts.csswg.org/css-page-3/#margin-boxes
+@page {
+ size: A4;
+ orientation: portrait;
+ padding: 0;
+ margin: 0;
+ margin-bottom: $a4PageMarginBottom;
+}
+
+@media screen, projection {
+ .printFooter, .printHeader {
+ display: none;
+ }
+}
+
+@media print {
+
+ html {
+ background-color: #FFF;
+ }
+
+ // When printing it is good practice to show the full url
+ a[href]:after {
+ content: ' [' attr(href) ']';
+ }
+
+ figure {
+ break-inside: avoid;
+ }
+
+ * {
+ -webkit-print-color-adjust: exact !important; /* Chrome, Safari, Edge */
+ color-adjust: exact !important; /*Firefox*/
+ }
+
+ .printHeader {
+ display: table-header-group;
+ position: fixed;
+ top: 0;
+ left: 50%;
+ padding-top: 2mm;
+ transform: translateX(-50%);
+ }
+
+ .printFooter {
+ display: table-footer-group;
+ position: fixed;
+ bottom: 0;
+ margin: 0 2mm 2mm 5mm;
+ > img {
+ height: 8mm;
+ }
+ }
+
+ // There is a known limitation that Chrome does not increment the counter of fixed elements even though
+ // they appear for each page. So we leave this out for now.
+ // See https://bugs.chromium.org/p/chromium/issues/detail?id=774830
+ // .printFooter::after {
+ // position: fixed;
+ // bottom: 2mm;
+ // left: 50%;
+ // transform: translateX(-50%);
+ // content: 'Page ' counter(x);
+ // }
+}
diff --git a/src/plugins/dashboard/public/application/embeddable/viewport/dashboard_viewport.tsx b/src/plugins/dashboard/public/application/embeddable/viewport/dashboard_viewport.tsx
index 0e700e058eef4..9c298fbf9be19 100644
--- a/src/plugins/dashboard/public/application/embeddable/viewport/dashboard_viewport.tsx
+++ b/src/plugins/dashboard/public/application/embeddable/viewport/dashboard_viewport.tsx
@@ -20,6 +20,8 @@ import {
} from '../../../../../controls/public';
import { withSuspense } from '../../../services/presentation_util';
+import { SharedPocPrintUi } from './shared_poc_print_ui';
+
export interface DashboardViewportProps {
container: DashboardContainer;
controlGroup?: ControlGroupContainer;
@@ -147,6 +149,7 @@ export class DashboardViewport extends React.Component}
+
>
);
}
diff --git a/src/plugins/dashboard/public/application/embeddable/viewport/shared_poc_print_ui.tsx b/src/plugins/dashboard/public/application/embeddable/viewport/shared_poc_print_ui.tsx
new file mode 100644
index 0000000000000..2f3b77e32c43f
--- /dev/null
+++ b/src/plugins/dashboard/public/application/embeddable/viewport/shared_poc_print_ui.tsx
@@ -0,0 +1,38 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License
+ * 2.0 and the Server Side Public License, v 1; you may not use this file except
+ * in compliance with, at your election, the Elastic License 2.0 or the Server
+ * Side Public License, v 1.
+ */
+
+import type { FunctionComponent } from 'react';
+import React from 'react';
+
+import { imgData } from './shared_poc_print_ui_logo';
+
+interface Props {
+ title: string;
+ logo?: string;
+}
+
+export const SharedPocPrintUi: FunctionComponent = ({ title, logo = imgData }) => {
+ return (
+ <>
+ {/* NOTE: This UI is purely for test purposes, but it is easy to see how we could move this to some external place that shares this functionality. */}
+
+
+ >
+ );
+};
diff --git a/src/plugins/dashboard/public/application/embeddable/viewport/shared_poc_print_ui_logo.tsx b/src/plugins/dashboard/public/application/embeddable/viewport/shared_poc_print_ui_logo.tsx
new file mode 100644
index 0000000000000..9356ccfa5692f
--- /dev/null
+++ b/src/plugins/dashboard/public/application/embeddable/viewport/shared_poc_print_ui_logo.tsx
@@ -0,0 +1,9 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License
+ * 2.0 and the Server Side Public License, v 1; you may not use this file except
+ * in compliance with, at your election, the Elastic License 2.0 or the Server
+ * Side Public License, v 1.
+ */
+
+export const imgData = ``;
diff --git a/src/plugins/dashboard/public/application/hooks/use_dashboard_app_state.ts b/src/plugins/dashboard/public/application/hooks/use_dashboard_app_state.ts
index a103c88843664..50c40e4863bee 100644
--- a/src/plugins/dashboard/public/application/hooks/use_dashboard_app_state.ts
+++ b/src/plugins/dashboard/public/application/hooks/use_dashboard_app_state.ts
@@ -183,8 +183,7 @@ export const useDashboardAppState = ({
savedDashboard,
});
- // Backwards compatible way of detecting that we are taking a screenshot
- const legacyPrintLayoutDetected =
+ const printLayoutDetected =
screenshotModeService?.isScreenshotMode() &&
screenshotModeService.getScreenshotContext('layout') === 'print';
@@ -194,8 +193,7 @@ export const useDashboardAppState = ({
...initialDashboardStateFromUrl,
...forwardedAppState,
- // if we are in legacy print mode, dashboard needs to be in print viewMode
- ...(legacyPrintLayoutDetected ? { viewMode: ViewMode.PRINT } : {}),
+ ...(printLayoutDetected ? { viewMode: ViewMode.PRINT } : {}),
// if there is an incoming embeddable, dashboard always needs to be in edit mode to receive it.
...(incomingEmbeddable ? { viewMode: ViewMode.EDIT } : {}),
diff --git a/x-pack/plugins/maps/kibana.json b/x-pack/plugins/maps/kibana.json
index a19f160127eb3..a17338d55de56 100644
--- a/x-pack/plugins/maps/kibana.json
+++ b/x-pack/plugins/maps/kibana.json
@@ -26,7 +26,8 @@
"savedObjects",
"share",
"presentationUtil",
- "sharedUX"
+ "sharedUX",
+ "screenshotMode"
],
"optionalPlugins": [
"cloud",
diff --git a/x-pack/plugins/maps/public/plugin.ts b/x-pack/plugins/maps/public/plugin.ts
index 2ab91462cfe7e..0145971f96500 100644
--- a/x-pack/plugins/maps/public/plugin.ts
+++ b/x-pack/plugins/maps/public/plugin.ts
@@ -80,6 +80,7 @@ import type { LensPublicSetup } from '../../lens/public';
import { setupLensChoroplethChart } from './lens';
import { SharedUXPluginStart } from '../../../../src/plugins/shared_ux/public';
+import { ScreenshotModePluginSetup } from '../../../../src/plugins/screenshot_mode/public';
export interface MapsPluginSetupDependencies {
cloud?: CloudSetup;
@@ -92,6 +93,7 @@ export interface MapsPluginSetupDependencies {
share: SharePluginSetup;
licensing: LicensingPluginSetup;
usageCollection?: UsageCollectionSetup;
+ screenshotMode: ScreenshotModePluginSetup;
}
export interface MapsPluginStartDependencies {
@@ -149,7 +151,12 @@ export class MapsPlugin
registerLicensedFeatures(plugins.licensing);
const config = this._initializerContext.config.get();
- setMapAppConfig(config);
+ setMapAppConfig({
+ ...config,
+ preserveDrawingBuffer: plugins.screenshotMode.isScreenshotMode()
+ ? true
+ : config.preserveDrawingBuffer,
+ });
const locator = plugins.share.url.locators.create(
new MapsAppLocatorDefinition({
diff --git a/x-pack/plugins/maps/tsconfig.json b/x-pack/plugins/maps/tsconfig.json
index 5d5f4223fab9a..57cc09dec4b16 100644
--- a/x-pack/plugins/maps/tsconfig.json
+++ b/x-pack/plugins/maps/tsconfig.json
@@ -33,6 +33,7 @@
{ "path": "../../../src/plugins/kibana_react/tsconfig.json" },
{ "path": "../../../src/plugins/kibana_utils/tsconfig.json" },
{ "path": "../../../src/plugins/shared_ux/tsconfig.json" },
+ { "path": "../../../src/plugins/screenshot_mode/tsconfig.json" },
{ "path": "../cloud/tsconfig.json" },
{ "path": "../features/tsconfig.json" },
{ "path": "../lens/tsconfig.json" },
diff --git a/x-pack/plugins/screenshotting/server/browsers/chromium/driver.ts b/x-pack/plugins/screenshotting/server/browsers/chromium/driver.ts
index f9bf31ad35f6f..390917db227be 100644
--- a/x-pack/plugins/screenshotting/server/browsers/chromium/driver.ts
+++ b/x-pack/plugins/screenshotting/server/browsers/chromium/driver.ts
@@ -155,6 +155,16 @@ export class HeadlessChromiumDriver {
return !this.page.isClosed();
}
+ async printA4Pdf(): Promise {
+ return this.page.pdf({
+ format: 'a4',
+ preferCSSPageSize: true,
+ scale: 1,
+ landscape: false,
+ displayHeaderFooter: true,
+ });
+ }
+
/*
* Call Page.screenshot and return a base64-encoded string of the image
*/
diff --git a/x-pack/plugins/screenshotting/server/formats/pdf/index.ts b/x-pack/plugins/screenshotting/server/formats/pdf/index.ts
index b7718155c5424..00112e13bbe66 100644
--- a/x-pack/plugins/screenshotting/server/formats/pdf/index.ts
+++ b/x-pack/plugins/screenshotting/server/formats/pdf/index.ts
@@ -99,28 +99,46 @@ export async function toPdf(
{ metrics, results }: CaptureResult
): Promise {
const timeRange = getTimeRange(results);
- try {
- const { buffer, pages } = await pngsToPdf({
- title: title ? `${title}${timeRange ? ` - ${timeRange}` : ''}` : undefined,
- results,
- layout,
- logo,
- packageInfo,
- logger,
- });
-
- return {
- metrics: {
- ...(metrics ?? {}),
- pages,
- },
- data: buffer,
- errors: results.flatMap(({ error }) => (error ? [error] : [])),
- renderErrors: results.flatMap(({ renderErrors }) => renderErrors ?? []),
- };
- } catch (error) {
- logger.error(`Could not generate the PDF buffer!`);
-
- throw error;
+ let buffer: Buffer;
+ let pages: number;
+ const shouldConvertPngsToPdf = layout.id !== LayoutTypes.PRINT;
+ if (shouldConvertPngsToPdf) {
+ try {
+ ({ buffer, pages } = await pngsToPdf({
+ title: title ? `${title}${timeRange ? ` - ${timeRange}` : ''}` : undefined,
+ results,
+ layout,
+ logo,
+ packageInfo,
+ logger,
+ }));
+
+ return {
+ metrics: {
+ ...(metrics ?? {}),
+ pages,
+ },
+ data: buffer,
+ errors: results.flatMap(({ error }) => (error ? [error] : [])),
+ renderErrors: results.flatMap(({ renderErrors }) => renderErrors ?? []),
+ };
+ } catch (error) {
+ logger.error(`Could not generate the PDF buffer!`);
+
+ throw error;
+ }
+ } else {
+ buffer = results[0].screenshots[0].data; // This buffer is already the PDF
+ pages = -1; // TODO: Figure out how to get page numbers
}
+
+ return {
+ metrics: {
+ ...(metrics ?? {}),
+ pages,
+ },
+ data: buffer,
+ errors: results.flatMap(({ error }) => (error ? [error] : [])),
+ renderErrors: results.flatMap(({ renderErrors }) => renderErrors ?? []),
+ };
}
diff --git a/x-pack/plugins/screenshotting/server/screenshots/get_pdf.ts b/x-pack/plugins/screenshotting/server/screenshots/get_pdf.ts
new file mode 100644
index 0000000000000..9b9629c3ab9af
--- /dev/null
+++ b/x-pack/plugins/screenshotting/server/screenshots/get_pdf.ts
@@ -0,0 +1,25 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License
+ * 2.0; you may not use this file except in compliance with the Elastic License
+ * 2.0.
+ */
+
+import type { Logger } from 'src/core/server';
+import type { HeadlessChromiumDriver } from '../browsers';
+import type { Screenshot } from './types';
+
+export async function getPdf(
+ browser: HeadlessChromiumDriver,
+ logger: Logger
+): Promise {
+ logger.info('printing PDF');
+
+ return [
+ {
+ data: await browser.printA4Pdf(),
+ title: null,
+ description: null,
+ },
+ ];
+}
diff --git a/x-pack/plugins/screenshotting/server/screenshots/get_screenshots.ts b/x-pack/plugins/screenshotting/server/screenshots/get_screenshots.ts
index 26ef272e7f18e..6899b9328736d 100644
--- a/x-pack/plugins/screenshotting/server/screenshots/get_screenshots.ts
+++ b/x-pack/plugins/screenshotting/server/screenshots/get_screenshots.ts
@@ -9,23 +9,7 @@ import apm from 'elastic-apm-node';
import type { Logger } from 'src/core/server';
import type { HeadlessChromiumDriver } from '../browsers';
import type { ElementsPositionAndAttribute } from './get_element_position_data';
-
-export interface Screenshot {
- /**
- * Screenshot PNG image data.
- */
- data: Buffer;
-
- /**
- * Screenshot title.
- */
- title: string | null;
-
- /**
- * Screenshot description.
- */
- description: string | null;
-}
+import type { Screenshot } from './types';
export const getScreenshots = async (
browser: HeadlessChromiumDriver,
diff --git a/x-pack/plugins/screenshotting/server/screenshots/observable.ts b/x-pack/plugins/screenshotting/server/screenshots/observable.ts
index fbc147102e0af..ec08c61456c0d 100644
--- a/x-pack/plugins/screenshotting/server/screenshots/observable.ts
+++ b/x-pack/plugins/screenshotting/server/screenshots/observable.ts
@@ -9,7 +9,7 @@ import type { Transaction } from 'elastic-apm-node';
import { defer, forkJoin, throwError, Observable } from 'rxjs';
import { catchError, mergeMap, switchMapTo, timeoutWith } from 'rxjs/operators';
import type { Headers, Logger } from 'src/core/server';
-import { errors } from '../../common';
+import { errors, LayoutTypes } from '../../common';
import type { Context, HeadlessChromiumDriver } from '../browsers';
import { getChromiumDisconnectedError, DEFAULT_VIEWPORT } from '../browsers';
import type { Layout } from '../layouts';
@@ -18,7 +18,8 @@ import { getElementPositionAndAttributes } from './get_element_position_data';
import { getNumberOfItems } from './get_number_of_items';
import { getRenderErrors } from './get_render_errors';
import { getScreenshots } from './get_screenshots';
-import type { Screenshot } from './get_screenshots';
+import { getPdf } from './get_pdf';
+import type { Screenshot } from './types';
import { getTimeRange } from './get_time_range';
import { injectCustomCss } from './inject_css';
import { openUrl } from './open_url';
@@ -247,7 +248,10 @@ export class ScreenshotObservableHandler {
getDefaultElementPosition(this.layout.getViewport(1));
let screenshots: Screenshot[] = [];
try {
- screenshots = await getScreenshots(this.driver, this.logger, elements);
+ screenshots =
+ this.layout.id === LayoutTypes.PRINT
+ ? await getPdf(this.driver, this.logger)
+ : await getScreenshots(this.driver, this.logger, elements);
} catch (e) {
throw new errors.FailedToCaptureScreenshot(e.message);
}
diff --git a/x-pack/plugins/screenshotting/server/screenshots/types.ts b/x-pack/plugins/screenshotting/server/screenshots/types.ts
new file mode 100644
index 0000000000000..d4a408313fc43
--- /dev/null
+++ b/x-pack/plugins/screenshotting/server/screenshots/types.ts
@@ -0,0 +1,23 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License
+ * 2.0; you may not use this file except in compliance with the Elastic License
+ * 2.0.
+ */
+
+export interface Screenshot {
+ /**
+ * Screenshot PNG image data.
+ */
+ data: Buffer;
+
+ /**
+ * Screenshot title.
+ */
+ title: string | null;
+
+ /**
+ * Screenshot description.
+ */
+ description: string | null;
+}