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. */} +
{title}
+
+ a cool logo for branding + {/* + Currently pageNumber is not being used, but the idea was to try and use JS to count the pages and populate + the repeated element, could not get this working + */} +
+
+ + ); +}; 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; +}