diff --git a/extensions/chromium/preferences_schema.json b/extensions/chromium/preferences_schema.json index eb23d18c0dfc08..99bf7dc872fc7a 100644 --- a/extensions/chromium/preferences_schema.json +++ b/extensions/chromium/preferences_schema.json @@ -67,6 +67,10 @@ "type": "boolean", "default": false }, + "enableFakeMLManager": { + "type": "boolean", + "default": false + }, "cursorToolOnLoad": { "title": "Cursor tool on load", "description": "The cursor tool that is enabled upon load.\n 0 = Text selection tool.\n 1 = Hand tool.", diff --git a/gulpfile.mjs b/gulpfile.mjs index 9385a4e8eaccb0..9fae8db6325aa0 100644 --- a/gulpfile.mjs +++ b/gulpfile.mjs @@ -201,6 +201,7 @@ function createWebpackAlias(defines) { "web-annotation_editor_params": "web/annotation_editor_params.js", "web-download_manager": "", "web-external_services": "", + "web-new_alt_text_manager": "web/new_alt_text_manager.js", "web-null_l10n": "", "web-pdf_attachment_viewer": "web/pdf_attachment_viewer.js", "web-pdf_cursor_tools": "web/pdf_cursor_tools.js", diff --git a/l10n/en-US/viewer.ftl b/l10n/en-US/viewer.ftl index 8aea43959e0bd5..534757460bb088 100644 --- a/l10n/en-US/viewer.ftl +++ b/l10n/en-US/viewer.ftl @@ -416,3 +416,52 @@ pdfjs-editor-colorpicker-red = pdfjs-editor-highlight-show-all-button-label = Show all pdfjs-editor-highlight-show-all-button = .title = Show all + +## New alt-text dialog +## Group note for entire feature: Alternative text (alt text) helps when people can't see the image. This feature includes a tool to create alt text automatically using an AI model that works locally on the user's device to preserve privacy. + +# Modal header positioned above a text box where users can edit the alt text. +pdfjs-editor-new-alt-text-dialog-edit-label = Edit alt text (image description) + +# Modal header positioned above a text box where users can edit the alt text. +pdfjs-editor-new-alt-text-dialog-add-label = Add alt text (image description) + +pdfjs-editor-new-alt-text-textarea = + .placeholder = Write your description here… + +# This text is refers to the alt text box above this description. It offers a definition of alt text. +pdfjs-editor-new-alt-text-description = Short description for people who can’t see the image or when the image doesn’t load. + +# This is a required legal disclaimer that refers to the automatically created text inside the alt text box above this text. It disappears if the text is edited by a human. +pdfjs-editor-new-alt-text-disclaimer = This alt text was created automatically. +pdfjs-editor-new-alt-text-disclaimer-learn-more-url = Learn more + +pdfjs-new-alt-text-create-automatically-button-label = Create alt text automatically +pdfjs-editor-new-alt-text-not-now-button = Not now +pdfjs-editor-new-alt-text-error-title = Couldn’t create alt text automatically +pdfjs-editor-new-alt-text-error-description = Please write your own alt text or try again later. +pdfjs-editor-new-alt-text-error-close-button = Close + +# Variables: +# $totalSize (Number) - the total size (in MB) of the ai model. +# $downloadedSize (Number) - the downloaded size (in MB) of the ai model. +# $percent (Number) - the percentage of the downloaded size. +pdfjs-editor-new-alt-text-ai-model-downloading-progress = + .aria-valuemin = 0 + .aria-valuemax = { $totalSize } + .aria-valuenow = { $downloadedSize } + .aria-valuetext = Downloading alt text AI model ({ $downloadedSize } of { $totalSize } MB) + +# This is a button that users can click to edit the alt text they have already added. +pdfjs-editor-new-alt-text-added-button-label = Alt text added + +# This is a button that users can click to open the alt text editor and add alt text when it is not present. +pdfjs-editor-new-alt-text-missing-button-label = Missing alt text + +# This is a button that opens up the alt text modal where users should review the alt text that was automatically generated. +pdfjs-editor-new-alt-text-to-review-button-label = Review alt text + +# "Created automatically" is a prefix that will be added to the beginning of any alt text that has been automatically generated. After the colon, the user will see/hear the actual alt text description. If the alt text has been edited by a human, this prefix will not appear. +# Variables: +# $generatedAltText (String) - the generated alt-text. +pdfjs-editor-new-alt-text-generated-alt-text-with-disclaimer = Created automatically: { $generatedAltText } diff --git a/src/display/editor/alt_text.js b/src/display/editor/alt_text.js index fececf29269cff..e5699e12acf1be 100644 --- a/src/display/editor/alt_text.js +++ b/src/display/editor/alt_text.js @@ -28,12 +28,23 @@ class AltText { #altTextWasFromKeyBoard = false; + #badge = null; + #editor = null; + #guessedText = ""; + + #amendedText = ""; + + #hasAI = false; + + #useNewAltTextFlow = false; + static _l10nPromise = null; constructor(editor) { this.#editor = editor; + this.#useNewAltTextFlow = editor._uiManager.useNewAltTextFlow; } static initialize(l10nPromise) { @@ -43,9 +54,17 @@ class AltText { async render() { const altText = (this.#altTextButton = document.createElement("button")); altText.className = "altText"; - const msg = await AltText._l10nPromise.get( - "pdfjs-editor-alt-text-button-label" - ); + let msg; + if (this.#useNewAltTextFlow) { + altText.classList.add("new"); + msg = await AltText._l10nPromise.get( + `pdfjs-editor-new-alt-text-${this.#hasAI ? "to-review" : "missing"}-button-label` + ); + } else { + msg = await AltText._l10nPromise.get( + "pdfjs-editor-alt-text-button-label" + ); + } altText.textContent = msg; altText.setAttribute("aria-label", msg); altText.tabIndex = "0"; @@ -87,6 +106,47 @@ class AltText { return !this.#altText && !this.#altTextDecorative; } + get guessedText() { + return this.#guessedText; + } + + async setGuessedText(guessedText) { + this.#guessedText = guessedText; + this.#amendedText = await AltText._l10nPromise.get( + "pdfjs-editor-new-alt-text-generated-alt-text-with-disclaimer" + )({ generatedAltText: guessedText }); + } + + toggleAltTextBadge(visibility) { + if (!this.#useNewAltTextFlow || this.#altText) { + if (this.#badge) { + this.#badge.remove(); + this.#badge = null; + } + return; + } + if (!this.#badge) { + const badge = (this.#badge = document.createElement("div")); + badge.className = "noAltTextBadge"; + this.#editor.div.append(badge); + } + this.#badge.classList.toggle("hidden", !visibility); + } + + serialize(isForCopying) { + let altText = this.#altText; + if (!isForCopying && this.#guessedText === altText) { + altText = this.#amendedText; + } + return { + altText, + decorative: this.#altTextDecorative, + guessedText: this.#guessedText, + amendedText: this.#amendedText, + hasAI: this.#hasAI, + }; + } + get data() { return { altText: this.#altText, @@ -97,12 +157,30 @@ class AltText { /** * Set the alt text data. */ - set data({ altText, decorative }) { - if (this.#altText === altText && this.#altTextDecorative === decorative) { + set data({ + altText, + decorative, + guessedText, + amendedText, + cancel = false, + hasAI = false, + }) { + if (guessedText) { + this.#guessedText = guessedText; + this.#amendedText = amendedText; + } + if ( + this.#altText === altText && + this.#altTextDecorative === decorative && + this.#hasAI === hasAI + ) { return; } - this.#altText = altText; - this.#altTextDecorative = decorative; + if (!cancel) { + this.#altText = altText; + this.#altTextDecorative = decorative; + } + this.#hasAI = hasAI; this.#setState(); } @@ -121,6 +199,8 @@ class AltText { this.#altTextButton?.remove(); this.#altTextButton = null; this.#altTextTooltip = null; + this.#badge?.remove(); + this.#badge = null; } async #setState() { @@ -128,18 +208,44 @@ class AltText { if (!button) { return; } - if (!this.#altText && !this.#altTextDecorative) { - button.classList.remove("done"); - this.#altTextTooltip?.remove(); - return; + + if (this.#useNewAltTextFlow) { + let type = "added"; + if (!this.#altText) { + type = this.#hasAI ? "to-review" : "missing"; + } + button.classList.toggle("done", type === "added"); + AltText._l10nPromise + .get(`pdfjs-editor-new-alt-text-${type}-button-label`) + .then(msg => { + button.setAttribute("aria-label", msg); + // We can't just use button.textContent here, because it would remove + // the existing tooltip element. + for (const child of button.childNodes) { + if (child.nodeType === Node.TEXT_NODE) { + child.textContent = msg; + break; + } + } + }); + if (!this.#altText) { + this.#altTextTooltip?.remove(); + return; + } + } else { + if (!this.#altText && !this.#altTextDecorative) { + button.classList.remove("done"); + this.#altTextTooltip?.remove(); + return; + } + button.classList.add("done"); + AltText._l10nPromise + .get("pdfjs-editor-alt-text-edit-button-label") + .then(msg => { + button.setAttribute("aria-label", msg); + }); } - button.classList.add("done"); - AltText._l10nPromise - .get("pdfjs-editor-alt-text-edit-button-label") - .then(msg => { - button.setAttribute("aria-label", msg); - }); let tooltip = this.#altTextTooltip; if (!tooltip) { this.#altTextTooltip = tooltip = document.createElement("span"); diff --git a/src/display/editor/editor.js b/src/display/editor/editor.js index 69dee6f15cfb7d..da38fc335d58c2 100644 --- a/src/display/editor/editor.js +++ b/src/display/editor/editor.js @@ -210,6 +210,9 @@ class AnnotationEditor { "pdfjs-editor-alt-text-button-label", "pdfjs-editor-alt-text-edit-button-label", "pdfjs-editor-alt-text-decorative-tooltip", + "pdfjs-editor-new-alt-text-added-button-label", + "pdfjs-editor-new-alt-text-missing-button-label", + "pdfjs-editor-new-alt-text-to-review-button-label", "pdfjs-editor-resizer-label-topLeft", "pdfjs-editor-resizer-label-topMiddle", "pdfjs-editor-resizer-label-topRight", @@ -223,6 +226,15 @@ class AnnotationEditor { l10n.get(str.replaceAll(/([A-Z])/g, c => `-${c.toLowerCase()}`)), ]) ); + + AnnotationEditor._l10nPromise.set( + "pdfjs-editor-new-alt-text-generated-alt-text-with-disclaimer", + l10n.get.bind( + l10n, + "pdfjs-editor-new-alt-text-generated-alt-text-with-disclaimer" + ) + ); + if (options?.strings) { for (const str of options.strings) { AnnotationEditor._l10nPromise.set(str, l10n.get(str)); @@ -1016,6 +1028,18 @@ class AnnotationEditor { this.#altText.data = data; } + get guessedAltText() { + return this.#altText.guessedText; + } + + async setGuessedAltText(text) { + await this.#altText.setGuessedText(text); + } + + serializeAltText(isForCopying) { + return this.#altText?.serialize(isForCopying); + } + hasAltText() { return !this.#altText?.isEmpty(); } @@ -1570,6 +1594,7 @@ class AnnotationEditor { return; } this.#editToolbar?.show(); + this.#altText?.toggleAltTextBadge(false); } /** @@ -1586,6 +1611,7 @@ class AnnotationEditor { }); } this.#editToolbar?.hide(); + this.#altText?.toggleAltTextBadge(true); } /** diff --git a/src/display/editor/stamp.js b/src/display/editor/stamp.js index c84078336e784f..8d86c2f06e28d9 100644 --- a/src/display/editor/stamp.js +++ b/src/display/editor/stamp.js @@ -36,8 +36,6 @@ class StampEditor extends AnnotationEditor { #canvas = null; - #hasMLBeenQueried = false; - #observer = null; #resizeTimeoutId = null; @@ -117,7 +115,12 @@ class StampEditor extends AnnotationEditor { #getBitmapDone() { this.#bitmapPromise = null; this._uiManager.enableWaiting(false); - if (this.#canvas) { + if (!this.#canvas) { + return; + } + if (this._uiManager.useNewAltTextFlow && this.#bitmap) { + this._uiManager.editAltText(this, /* firstTime = */ true); + } else { this.div.focus(); } } @@ -348,6 +351,43 @@ class StampEditor extends AnnotationEditor { } } + copyCanvas(maxDimension) { + const { width: bitmapWidth, height: bitmapHeight } = this.#bitmap; + const canvas = document.createElement("canvas"); + + let bitmap = this.#bitmap; + let width = bitmapWidth, + height = bitmapHeight; + if (bitmapWidth > maxDimension || bitmapHeight > maxDimension) { + const ratio = Math.min( + maxDimension / bitmapWidth, + maxDimension / bitmapHeight + ); + width = Math.floor(bitmapWidth * ratio); + height = Math.floor(bitmapHeight * ratio); + + if (!this.#isSvg) { + bitmap = this.#scaleBitmap(width, height); + } + } + canvas.width = width; + canvas.height = height; + const ctx = canvas.getContext("2d", { willReadFrequently: true }); + ctx.filter = this._uiManager.hcmFilter; + ctx.drawImage( + bitmap, + 0, + 0, + bitmap.width, + bitmap.height, + 0, + 0, + width, + height + ); + return canvas; + } + /** * When the dimensions of the div change the inner canvas must * renew its dimensions, hence it must redraw its own contents. @@ -425,43 +465,6 @@ class StampEditor extends AnnotationEditor { return bitmap; } - async #mlGuessAltText(bitmap, width, height) { - if (this.#hasMLBeenQueried) { - return; - } - this.#hasMLBeenQueried = true; - const isMLEnabled = await this._uiManager.isMLEnabledFor("altText"); - if (!isMLEnabled || this.hasAltText()) { - return; - } - const offscreen = new OffscreenCanvas(width, height); - const ctx = offscreen.getContext("2d", { willReadFrequently: true }); - ctx.drawImage( - bitmap, - 0, - 0, - bitmap.width, - bitmap.height, - 0, - 0, - width, - height - ); - const response = await this._uiManager.mlGuess({ - service: "moz-image-to-text", - request: { - data: ctx.getImageData(0, 0, width, height).data, - width, - height, - channels: 4, - }, - }); - const altText = response?.output || ""; - if (this.parent && altText && !this.hasAltText()) { - this.altTextData = { altText, decorative: false }; - } - } - #drawBitmap(width, height) { width = Math.ceil(width); height = Math.ceil(height); @@ -475,8 +478,6 @@ class StampEditor extends AnnotationEditor { ? this.#bitmap : this.#scaleBitmap(width, height); - this.#mlGuessAltText(bitmap, width, height); - const ctx = canvas.getContext("2d"); ctx.filter = this._uiManager.hcmFilter; ctx.drawImage( @@ -616,11 +617,11 @@ class StampEditor extends AnnotationEditor { // of this annotation and the clipboard doesn't support ImageBitmaps, // hence we serialize the bitmap to a data url. serialized.bitmapUrl = this.#serializeBitmap(/* toUrl = */ true); - serialized.accessibilityData = this.altTextData; + serialized.accessibilityData = this.serializeAltText(true); return serialized; } - const { decorative, altText } = this.altTextData; + const { decorative, altText } = this.serializeAltText(false); if (!decorative && altText) { serialized.accessibilityData = { type: "Figure", alt: altText }; } diff --git a/src/display/editor/tools.js b/src/display/editor/tools.js index fa4be1fd1ea60b..585634941eae9a 100644 --- a/src/display/editor/tools.js +++ b/src/display/editor/tools.js @@ -851,6 +851,10 @@ class AnnotationEditorUIManager { } } + hasMLManager() { + return !!this.#mlManager; + } + async mlGuess(data) { return this.#mlManager?.guess(data) || null; } @@ -859,6 +863,10 @@ class AnnotationEditorUIManager { return !!(await this.#mlManager?.isEnabledFor(name)); } + get mlManager() { + return this.#mlManager; + } + get useNewAltTextFlow() { return this.#enableUpdatedAddImage; } @@ -912,8 +920,8 @@ class AnnotationEditorUIManager { this.#mainHighlightColorPicker = colorPicker; } - editAltText(editor) { - this.#altTextManager?.editAltText(this, editor); + editAltText(editor, firstTime = false) { + this.#altTextManager?.editAltText(this, editor, firstTime); } switchToMode(mode, callback) { diff --git a/web/annotation_editor_layer_builder.css b/web/annotation_editor_layer_builder.css index eb13c7e46c05e8..cf854b6702d27e 100644 --- a/web/annotation_editor_layer_builder.css +++ b/web/annotation_editor_layer_builder.css @@ -46,6 +46,8 @@ --editorFreeHighlight-editing-cursor: url(images/cursor-editorFreeHighlight.svg) 1 18, pointer; + + --new-alt-text-warning-image: url(images/altText_warning.svg); } /* The following class is used to hide an element but keep it available to @@ -222,12 +224,18 @@ --editor-toolbar-vert-offset: 6px; --editor-toolbar-height: 28px; --editor-toolbar-padding: 2px; + --alt-text-done-color: #2ac3a2; + --alt-text-warning-color: #0090ed; + --alt-text-hover-done-color: var(--alt-text-done-color); + --alt-text-hover-warning-color: var(--alt-text-warning-color); @media (prefers-color-scheme: dark) { --editor-toolbar-bg-color: #2b2a33; --editor-toolbar-fg-color: #fbfbfe; --editor-toolbar-hover-bg-color: #52525e; --editor-toolbar-focus-outline-color: #0df; + --alt-text-done-color: #54ffbd; + --alt-text-warning-color: #80ebff; } @media screen and (forced-colors: active) { @@ -241,6 +249,10 @@ var(--editor-toolbar-hover-border-color); --editor-toolbar-focus-outline-color: ButtonBorder; --editor-toolbar-shadow: none; + --alt-text-done-color: var(--editor-toolbar-fg-color); + --alt-text-warning-color: var(--editor-toolbar-fg-color); + --alt-text-hover-done-color: var(--editor-toolbar-hover-fg-color); + --alt-text-hover-warning-color: var(--editor-toolbar-hover-fg-color); } display: flex; @@ -400,6 +412,30 @@ mask-image: var(--alt-text-done-image); } + &.new { + &::before { + width: 16px; + height: 16px; + mask-image: var(--new-alt-text-warning-image); + background-color: var(--alt-text-warning-color); + } + + &:hover::before { + background-color: var(--alt-text-hover-warning-color); + } + + &.done { + &:hover::before { + background-color: var(--alt-text-hover-done-color); + } + + &::before { + mask-image: var(--alt-text-done-image); + background-color: var(--alt-text-done-color); + } + } + } + .tooltip { display: none; @@ -519,6 +555,49 @@ top: 0; left: 0; } + + .noAltTextBadge { + --no-alt-text-badge-border-color: #f0f0f4; + --no-alt-text-badge-bg-color: #cfcfd8; + --no-alt-text-badge-fg-color: #5b5b66; + + @media (prefers-color-scheme: dark) { + --no-alt-text-badge-border-color: #52525e; + --no-alt-text-badge-bg-color: #fbfbfe; + --no-alt-text-badge-fg-color: #15141a; + } + + @media screen and (forced-colors: active) { + --no-alt-text-badge-border-color: ButtonText; + --no-alt-text-badge-bg-color: ButtonFace; + --no-alt-text-badge-fg-color: ButtonText; + } + + position: absolute; + inset-inline-end: 5px; + inset-block-end: 5px; + display: inline-flex; + width: 32px; + height: 32px; + padding: 3px; + justify-content: center; + align-items: center; + pointer-events: none; + + border-radius: 2px; + border: 1px solid var(--no-alt-text-badge-border-color); + background: var(--no-alt-text-badge-bg-color); + + &::before { + content: ""; + display: inline-block; + width: 16px; + height: 16px; + mask-image: var(--new-alt-text-warning-image); + mask-size: cover; + background-color: var(--no-alt-text-badge-fg-color); + } + } } .annotationEditorLayer { @@ -767,6 +846,204 @@ } } +.dialog.newAltText { + --new-alt-text-ai-disclaimer-icon: url(images/altText_disclaimer.svg); + --new-alt-text-spinner-icon: url(images/altText_spinner.svg); + + width: 80%; + max-width: 570px; + min-width: 300px; + padding: 0; + + &.aiDisabled { + #newAltTextDisclaimer { + display: none !important; + } + + #newAltTextDescriptionTextarea::placeholder { + color: var(--text-secondary-color) !important; + } + } + + &.edited { + #newAltTextDisclaimer { + display: none !important; + } + #newAltTextDescriptionTextarea::placeholder { + color: var(--text-secondary-color) !important; + } + } + + &.noAi { + #newAltTextDisclaimer, + #newAltTextCreateAutomatically { + display: none !important; + } + #newAltTextDescriptionTextarea::placeholder { + color: var(--text-secondary-color) !important; + } + } + + &.aiInstalling { + #newAltTextCreateAutomatically { + display: none !important; + } + #newAltTextDownloadModel { + display: flex !important; + } + #newAltTextDescriptionTextarea::placeholder { + color: var(--text-secondary-color) !important; + } + } + + &.error { + #newAltTextDisclaimer { + display: none !important; + } + + #newAltTextDescriptionTextarea::placeholder { + color: var(--text-secondary-color) !important; + } + + #newAltTextNotNow { + display: none !important; + } + + #newAltTextCancel { + display: inline-block !important; + } + } + + #newAltTextContainer { + display: flex; + width: auto; + padding: 16px; + flex-direction: column; + justify-content: flex-end; + align-items: flex-start; + gap: 12px; + flex: 0 1 auto; + + #mainContent { + display: flex; + justify-content: flex-end; + align-items: flex-start; + gap: 12px; + align-self: stretch; + flex: 1 1 auto; + + #functions { + display: flex; + flex-direction: column; + align-items: flex-start; + gap: 16px; + flex: 1 0 0; + align-self: stretch; + } + + #descriptionInstruction { + display: flex; + flex-direction: column; + align-items: flex-start; + gap: 8px; + align-self: stretch; + flex: 1 1 auto; + + #newAltTextDescriptionContainer { + width: 100%; + height: 70px; + position: relative; + + textarea { + width: 100%; + height: 100%; + padding: 8px; + + &::placeholder { + color: transparent; + } + } + + .altTextSpinner { + display: none; + position: absolute; + width: 16px; + height: 16px; + inset-inline-start: 8px; + inset-block-start: 8px; + mask-size: cover; + background-color: var(--text-secondary-color); + pointer-events: none; + } + + &.loading .altTextSpinner { + display: inline-block; + mask-image: var(--new-alt-text-spinner-icon); + } + } + + #newAltTextDescription { + font-size: 11px; + } + + #newAltTextDisclaimer { + display: flex; + align-items: center; + gap: 4px; + align-self: stretch; + flex-wrap: wrap; + font-size: 11px; + + &::before { + content: ""; + display: inline-block; + width: 16px; + height: 16px; + mask-image: var(--new-alt-text-ai-disclaimer-icon); + mask-size: cover; + background-color: var(--text-secondary-color); + } + } + } + + #newAltTextDownloadModel { + display: flex; + align-items: center; + gap: 4px; + align-self: stretch; + + &::before { + content: ""; + display: inline-block; + width: 16px; + height: 16px; + mask-image: var(--new-alt-text-spinner-icon); + mask-size: cover; + background-color: var(--text-secondary-color); + } + } + + #newAltTextImagePreview { + width: 180px; + aspect-ratio: 1; + display: flex; + justify-content: center; + align-items: center; + flex: 0 0 auto; + + > canvas { + max-width: 100%; + max-height: 100%; + } + } + } + + #newAltTextButtons { + align-self: flex-end; + } + } +} + .colorPicker { --hover-outline-color: #0250bb; --selected-outline-color: #0060df; diff --git a/web/app.js b/web/app.js index e28b43b694bcfe..3aeb520877d105 100644 --- a/web/app.js +++ b/web/app.js @@ -64,6 +64,7 @@ import { AltTextManager } from "web-alt_text_manager"; import { AnnotationEditorParams } from "web-annotation_editor_params"; import { CaretBrowsingMode } from "./caret_browsing.js"; import { DownloadManager } from "web-download_manager"; +import { NewAltTextManager } from "web-new_alt_text_manager"; import { OverlayManager } from "./overlay_manager.js"; import { PasswordPrompt } from "./password_prompt.js"; import { PDFAttachmentViewer } from "web-pdf_attachment_viewer"; @@ -205,6 +206,12 @@ const PDFViewerApplication = { if (mode) { document.documentElement.classList.add(mode); } + if (AppOptions.get("enableFakeMLManager")) { + this.mlManager = + MLManager.getFakeMLManager?.({ + enableGuessAltText: AppOptions.get("enableGuessAltText"), + }) || null; + } } else if (AppOptions.get("enableAltText")) { // We want to load the image-to-text AI engine as soon as possible. this.mlManager = new MLManager({ @@ -433,14 +440,21 @@ const PDFViewerApplication = { foreground: AppOptions.get("pageColorsForeground"), } : null; - const altTextManager = appConfig.altTextDialog - ? new AltTextManager( - appConfig.altTextDialog, - container, - this.overlayManager, - eventBus - ) - : null; + let altTextManager; + if (AppOptions.get("enableUpdatedAddImage")) { + altTextManager = appConfig.newAltTextDialog + ? new NewAltTextManager(appConfig.newAltTextDialog, this.overlayManager) + : null; + } else { + altTextManager = appConfig.altTextDialog + ? new AltTextManager( + appConfig.altTextDialog, + container, + this.overlayManager, + eventBus + ) + : null; + } const enableHWA = AppOptions.get("enableHWA"); const pdfViewer = new PDFViewer({ diff --git a/web/app_options.js b/web/app_options.js index 7b685ff2841efe..f2b1b8bc2c1fc5 100644 --- a/web/app_options.js +++ b/web/app_options.js @@ -187,9 +187,14 @@ const defaultOptions = { }, enableAltText: { /** @type {boolean} */ - value: false, + value: true, kind: OptionKind.VIEWER + OptionKind.PREFERENCE, }, + enableFakeMLManager: { + /** @type {boolean} */ + value: true, + kind: OptionKind.VIEWER, + }, enableGuessAltText: { /** @type {boolean} */ value: true, @@ -231,7 +236,7 @@ const defaultOptions = { // in Firefox release, but it has to be temporary. // TODO: remove it when unnecessary. /** @type {boolean} */ - value: false, + value: true, kind: OptionKind.VIEWER + OptionKind.PREFERENCE, }, externalLinkRel: { diff --git a/web/dialog.css b/web/dialog.css index 13a78900f15d62..8008e8a42955a2 100644 --- a/web/dialog.css +++ b/web/dialog.css @@ -22,6 +22,8 @@ --hover-filter: brightness(0.9); --focus-ring-color: #0060df; --focus-ring-outline: 2px solid var(--focus-ring-color); + --link-fg-color: #0060df; + --link-hover-fg-color: #0250bb; --textarea-border-color: #8f8f9d; --textarea-bg-color: white; @@ -53,6 +55,8 @@ --text-secondary-color: #cfcfd8; --focus-ring-color: #0df; --hover-filter: brightness(1.4); + --link-fg-color: #0df; + --link-hover-fg-color: #80ebff; --textarea-bg-color: #42414d; @@ -73,6 +77,8 @@ --text-secondary-color: CanvasText; --hover-filter: none; --focus-ring-color: ButtonBorder; + --link-fg-color: LinkText; + --link-hover-fg-color: LinkText; --textarea-border-color: ButtonBorder; --textarea-bg-color: Field; @@ -112,6 +118,22 @@ outline-offset: 2px; } + .title { + display: flex; + width: auto; + flex-direction: column; + justify-content: flex-end; + align-items: flex-start; + gap: 12px; + + > span { + font-size: 13px; + font-style: normal; + font-weight: 590; + line-height: 150%; /* 19.5px */ + } + } + .radio { display: flex; flex-direction: column; @@ -159,7 +181,36 @@ } } - button { + .checkbox { + display: flex; + align-items: center; + gap: 8px; + align-self: stretch; + + > .checkboxButton { + display: flex; + padding-top: 2px; + flex-direction: column; + align-items: flex-start; + width: 16px; + height: 16px; + + border-radius: 2px; + border: 1px solid var(--Light-Color-Border-Default, #8f8f9d); + background: var(--Light-Color-Action-Secondary-Default, #f0f0f4); + } + + > .checkboxLabel { + color: var(--Light-Color-Text-Primary, #15141a); + + font-size: 13px; + font-style: normal; + font-weight: 400; + line-height: normal; + } + } + + button:not(:is(.toggle-button, .closeButton)) { border-radius: 4px; border: 1px solid; font: menu; @@ -199,6 +250,14 @@ } } + a { + color: var(--link-fg-color); + + &:hover { + color: var(--link-hover-fg-color); + } + } + textarea { font: inherit; padding: 8px; @@ -220,5 +279,153 @@ opacity: 0.4; } } + + .messageBar { + --message-bar-warning-icon: url(images/messageBar_warning.svg); + --closing-button-icon: url(images/messageBar_closingButton.svg); + + --message-bar-bg-color: #ffebcd; + --message-bar-fg-color: #15141a; + --message-bar-border-color: rgb(0 0 0 / 0.08); + --message-bar-icon-color: #cd411e; + --message-bar-close-button-border-radius: 4px; + --message-bar-close-button-border: none; + --message-bar-close-button-color: var(--text-primary-color); + --message-bar-close-button-hover-bg-color: rgb(21 20 26 / 0.14); + --message-bar-close-button-active-bg-color: rgb(21 20 26 / 0.21); + --message-bar-close-button-focus-bg-color: rgb(21 20 26 / 0.07); + --message-bar-close-button-color-hover: var(--text-primary-color); + + @media (prefers-color-scheme: dark) { + --message-bar-bg-color: #5a3100; + --message-bar-fg-color: #fbfbfe; + --message-bar-border-color: rgb(255 255 255 / 0.08); + --message-bar-icon-color: #e49c49; + --message-bar-close-button-hover-bg-color: rgb(251 251 254 / 0.14); + --message-bar-close-button-active-bg-color: rgb(251 251 254 / 0.21); + --message-bar-close-button-focus-bg-color: rgb(251 251 254 / 0.07); + } + + @media screen and (forced-colors: active) { + --message-bar-bg-color: HighlightText; + --message-bar-fg-color: CanvasText; + --message-bar-border-color: CanvasText; + --message-bar-icon-color: CanvasText; + --message-bar-close-button-color: ButtonText; + --message-bar-close-button-border: 1px solid ButtonText; + --message-bar-close-button-hover-bg-color: ButtonText; + --message-bar-close-button-active-bg-color: ButtonText; + --message-bar-close-button-focus-bg-color: ButtonText; + --message-bar-close-button-color-hover: HighlightText; + } + + display: flex; + position: relative; + padding: 12px 8px 12px 0; + flex-direction: column; + justify-content: center; + align-items: flex-start; + gap: 8px; + align-self: stretch; + + border-radius: 4px; + border: 1px solid var(--message-bar-border-color); + background: var(--message-bar-bg-color); + + > div { + display: flex; + padding-inline-start: 16px; + align-items: flex-start; + gap: 8px; + align-self: stretch; + + &::before { + content: ""; + display: inline-block; + width: 16px; + height: 16px; + mask-image: var(--message-bar-warning-icon); + mask-size: cover; + background-color: var(--message-bar-icon-color); + } + + > div { + display: flex; + flex-direction: column; + align-items: flex-start; + gap: 8px; + flex: 1 0 0; + + .title { + font-size: 13px; + font-weight: 590; + } + + .description { + font-size: 13px; + } + } + } + + .closeButton { + position: absolute; + width: 32px; + height: 32px; + inset-inline-end: 8px; + inset-block-start: 8px; + background: none; + border-radius: var(--message-bar-close-button-border-radius); + border: var(--message-bar-close-button-border); + + &::before { + content: ""; + display: inline-block; + width: 16px; + height: 16px; + mask-image: var(--closing-button-icon); + mask-size: cover; + background-color: var(--message-bar-close-button-color); + } + + &:is(:hover, :active, :focus)::before { + background-color: var(--message-bar-close-button-color-hover); + } + + &:hover { + background-color: var(--message-bar-close-button-hover-bg-color); + } + + &:active { + background-color: var(--message-bar-close-button-active-bg-color); + } + + &:focus { + background-color: var(--message-bar-close-button-focus-bg-color); + } + + > span { + display: inline-block; + width: 0; + height: 0; + overflow: hidden; + } + } + } + + .toggler { + display: flex; + align-items: center; + gap: 8px; + align-self: stretch; + + > .togglerLabel { + color: var(--Light-Color-Text-Primary, #15141a); + + font-size: 13px; + font-style: normal; + font-weight: 400; + line-height: normal; + } + } } } diff --git a/web/firefoxcom.js b/web/firefoxcom.js index 869b69564ce06e..5c5a8e18b95804 100644 --- a/web/firefoxcom.js +++ b/web/firefoxcom.js @@ -310,6 +310,8 @@ class FirefoxScripting { class MLManager { #enabled = null; + #ready = null; + eventBus = null; constructor(options) { @@ -320,6 +322,10 @@ class MLManager { return !!(await this.#enabled?.get(name)); } + isReady(name) { + return this.#ready?.has(name) ?? false; + } + deleteModel(service) { return FirefoxCom.requestAsync("mlDelete", service); } @@ -338,14 +344,32 @@ class MLManager { this.altTextLearnMoreUrl = altTextLearnMoreUrl; } + async toggleService(name, enabled) { + if (name !== "altText") { + return; + } + + if (enabled) { + await this.#loadAltTextEngine(false); + } else { + this.#enabled.delete(name); + } + } + async #loadAltTextEngine(listenToProgress) { if (this.#enabled?.has("altText")) { // We already have a promise for the "altText" service. return; } + this.#ready ||= new Set(); const promise = FirefoxCom.requestAsync("loadAIEngine", { service: "moz-image-to-text", listenToProgress, + }).then(ok => { + if (ok) { + this.#ready.add("altText"); + } + return ok; }); (this.#enabled ||= new Map()).set("altText", promise); if (listenToProgress) { diff --git a/web/genericcom.js b/web/genericcom.js index dc3d60a98c878c..127b3699d46268 100644 --- a/web/genericcom.js +++ b/web/genericcom.js @@ -56,9 +56,50 @@ class MLManager { return null; } - async guess() { + isReady(_name) { + return false; + } + + guess() {} + + toggleService(_name, _enabled) {} + + static getFakeMLManager(options) { + return new FakeMLManager(options); + } +} + +class FakeMLManager { + constructor({ enableGuessAltText }) { + this.enableGuessAltText = enableGuessAltText; + } + + async isEnabledFor(_name) { + return this.enableGuessAltText; + } + + async deleteModel(_service) { return null; } + + isReady(_name) { + return true; + } + + guess({ request: { data } }) { + if (!data) { + return Promise.resolve({ error: true }); + } + return new Promise(resolve => { + setTimeout(() => { + resolve({ output: "Fake alt text" }); + }, 3000); + }); + } + + toggleService(_name, enabled) { + this.enableGuessAltText = enabled; + } } export { ExternalServices, initCom, MLManager, Preferences }; diff --git a/web/images/altText_disclaimer.svg b/web/images/altText_disclaimer.svg new file mode 100644 index 00000000000000..6fe79e710c82a0 --- /dev/null +++ b/web/images/altText_disclaimer.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/images/altText_spinner.svg b/web/images/altText_spinner.svg new file mode 100644 index 00000000000000..1a10324c5a487c --- /dev/null +++ b/web/images/altText_spinner.svg @@ -0,0 +1,16 @@ + + + + + + diff --git a/web/images/altText_warning.svg b/web/images/altText_warning.svg new file mode 100644 index 00000000000000..03014ceab2506c --- /dev/null +++ b/web/images/altText_warning.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/images/messageBar_closingButton.svg b/web/images/messageBar_closingButton.svg new file mode 100644 index 00000000000000..f2af2d14f62dd2 --- /dev/null +++ b/web/images/messageBar_closingButton.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/web/images/messageBar_warning.svg b/web/images/messageBar_warning.svg new file mode 100644 index 00000000000000..8d3c79abae9b81 --- /dev/null +++ b/web/images/messageBar_warning.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/web/new_alt_text_manager.js b/web/new_alt_text_manager.js new file mode 100644 index 00000000000000..9a83dd014f3b48 --- /dev/null +++ b/web/new_alt_text_manager.js @@ -0,0 +1,387 @@ +/* Copyright 2023 Mozilla Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { shadow } from "pdfjs-lib"; + +class NewAltTextManager { + #boundCancel = this.#cancel.bind(this); + + #createAutomaticallyButton; + + #currentEditor = null; + + #cancelButton; + + #descriptionContainer; + + #dialog; + + #error; + + #guessedAltText; + + #hasAI = false; + + #isEditing = false; + + #imagePreview; + + #isAILoading = false; + + #learnMore; + + #notNowButton; + + #overlayManager; + + #saveButton; + + #textarea; + + #title; + + #uiManager; + + #previousAltText = null; + + #telemetryData = null; + + constructor( + { + descriptionContainer, + dialog, + imagePreview, + cancelButton, + notNowButton, + saveButton, + textarea, + learnMore, + errorCloseButton, + error, + createAutomaticallyButton, + title, + }, + overlayManager + ) { + this.#cancelButton = cancelButton; + this.#createAutomaticallyButton = createAutomaticallyButton; + this.#descriptionContainer = descriptionContainer; + this.#dialog = dialog; + this.#error = error; + this.#notNowButton = notNowButton; + this.#saveButton = saveButton; + this.#imagePreview = imagePreview; + this.#textarea = textarea; + this.#learnMore = learnMore; + this.#title = title; + this.#overlayManager = overlayManager; + + dialog.addEventListener("close", this.#close.bind(this)); + dialog.addEventListener("contextmenu", event => { + if (event.target !== this.#textarea) { + event.preventDefault(); + } + }); + cancelButton.addEventListener("click", this.#boundCancel); + notNowButton.addEventListener("click", this.#boundCancel); + saveButton.addEventListener("click", this.#save.bind(this)); + errorCloseButton.addEventListener("click", () => { + error.classList.toggle("hidden", true); + }); + createAutomaticallyButton.addEventListener("click", async () => { + const checked = + createAutomaticallyButton.getAttribute("aria-pressed") !== "true"; + if (this.#uiManager) { + this.#uiManager.setPreference("enableGuessAltText", checked); + await this.#uiManager.mlManager.toggleService("altText", checked); + } + this.toggleGuessAltText(checked); + }); + textarea.addEventListener("focus", () => { + this.toggleLoading(false); + }); + textarea.addEventListener("blur", () => { + if (textarea.value) { + return; + } + this.toggleLoading(this.#isAILoading); + }); + textarea.addEventListener("input", () => { + this.toggleTitle(!!textarea.value); + this.toggleEdited(); + }); + + this.#overlayManager.register(dialog); + } + + get _elements() { + return shadow(this, "_elements", [ + this.#textarea, + this.#saveButton, + this.#cancelButton, + this.#notNowButton, + ]); + } + + toggleLoading(value) { + if (!this.#uiManager) { + return; + } + this.#descriptionContainer.classList.toggle("loading", value); + } + + toggleEdited() { + if (!this.#uiManager || this.#isAILoading) { + return; + } + this.#dialog.classList.toggle( + "edited", + !this.#guessedAltText || this.#textarea.value !== this.#guessedAltText + ); + } + + toggleError(value) { + if (!this.#uiManager) { + return; + } + this.#dialog.classList.toggle("error", value); + this.#error.classList.toggle("hidden", !value); + } + + toggleTitle(isEditing) { + if (this.#isEditing === isEditing) { + return; + } + this.#isEditing = isEditing; + this.#title.setAttribute( + "data-l10n-id", + `pdfjs-editor-new-alt-text-dialog-${isEditing ? "edit" : "add"}-label` + ); + } + + async toggleGuessAltText(value) { + if (!this.#uiManager) { + return; + } + this.#dialog.classList.toggle("aiDisabled", !value); + this.#createAutomaticallyButton.setAttribute("aria-pressed", value); + this.toggleTitle(value); + this.#hasAI = value; + + if (value) { + const { mlManager } = this.#uiManager; + const { altTextLearnMoreUrl } = mlManager; + if (altTextLearnMoreUrl) { + this.#learnMore.href = altTextLearnMoreUrl; + } + this.#guessedAltText = this.#currentEditor.guessedAltText; + this.#mlGuessAltText(); + } else { + this.#isAILoading = false; + this.toggleLoading(false); + this.#guessedAltText = ""; + } + this.toggleEdited(); + } + + toggleNotNow(firstTime) { + this.#notNowButton.classList.toggle("hidden", !firstTime); + this.#cancelButton.classList.toggle("hidden", firstTime); + } + + toggleAI(value) { + this.#dialog.classList.toggle("noAi", !value); + if (!value) { + this.toggleTitle(false); + } + } + + async #mlGuessAltText() { + if (!this.#hasAI || this.#isAILoading || this.#textarea.value) { + return; + } + let altText = this.#guessedAltText; + if (altText) { + this.#addAltText(altText); + return; + } + + let hasError = false; + try { + this.#isAILoading = true; + this.toggleLoading(true); + const canvas = this.#imagePreview.firstChild; + const { width, height } = canvas; + const ctx = canvas.getContext("2d"); + const { data } = ctx.getImageData(0, 0, width, height); + + // Take a reference on the current editor, as it can be set to null (if + // the dialog is closed before the end of the guess). + // But in case we've an alt-text, we want to set it on the editor. + const editor = this.#currentEditor; + + // When calling #mlGuessAltText we don't wait for it, so we must take care + // that the alt text dialog can have been closed before the response is. + const response = await this.#uiManager.mlGuess({ + service: "moz-image-to-text", + request: { + data, + width, + height, + channels: data.length / (width * height), + }, + }); + hasError = !response || response.error || !response.output; + if (!hasError) { + this.#guessedAltText = altText = response.output; + await editor.setGuessedAltText(altText); + if (this.#isAILoading) { + this.#addAltText(altText); + } + } + } catch (e) { + console.error(e); + hasError = true; + } finally { + this.#isAILoading = false; + this.toggleLoading(false); + } + + if (hasError && this.#uiManager) { + this.toggleError(true); + this.#title.setAttribute( + "data-l10n-id", + "pdfjs-editor-new-alt-text-dialog-add-label" + ); + this.#hasAI = false; + } + } + + #addAltText(altText) { + if (!this.#uiManager) { + return; + } + if (!this.#textarea.value) { + this.#textarea.value = altText; + this.altTextData = { altText, decorative: false }; + } + } + + async editAltText(uiManager, editor, firstTime) { + if (this.#currentEditor || !editor || (firstTime && editor.hasAltText())) { + return; + } + + let { mlManager } = uiManager; + if (!mlManager?.isReady("altText")) { + mlManager = null; + } + + // TODO: get this value from Firefox + // (https://bugzilla.mozilla.org/show_bug.cgi?id=1908184) + const AI_MAX_IMAGE_DIMENSION = 224; + + // The max dimension of the preview in the dialog is 180px, so we keep 224px + // and rescale it thanks to css. + const canvas = editor.copyCanvas(AI_MAX_IMAGE_DIMENSION); + canvas.setAttribute("role", "presentation"); + this.#imagePreview.append(canvas); + + this.#currentEditor = editor; + this.#uiManager = uiManager; + this.#uiManager.removeEditListeners(); + + const { altText } = editor.altTextData; + this.#previousAltText = this.#textarea.value = altText?.trim() || ""; + + this.#hasAI = false; + if (mlManager) { + this.toggleGuessAltText(await mlManager.isEnabledFor("altText")); + } + + this.toggleNotNow(firstTime); + this.toggleAI(!!mlManager); + this.toggleError(false); + + try { + await this.#overlayManager.open(this.#dialog); + } catch (ex) { + this.#close(); + throw ex; + } + } + + #cancel() { + this.#currentEditor.altTextData = { + cancel: true, + hasAI: this.#hasAI, + }; + this.#finish(); + } + + #finish() { + if (this.#overlayManager.active === this.#dialog) { + this.#overlayManager.close(this.#dialog); + } + } + + #close() { + this.#currentEditor._reportTelemetry( + this.#telemetryData || { + action: "alt_text_cancel", + } + ); + this.#telemetryData = null; + this.#isAILoading = false; + this.toggleLoading(false); + + this.#uiManager?.addEditListeners(); + this.#currentEditor.altTextFinish(); + this.#uiManager.setSelected(this.#currentEditor); + this.#currentEditor = null; + this.#uiManager = null; + const canvas = this.#imagePreview.firstChild; + if (canvas) { + canvas.width = canvas.height = 0; + canvas.remove(); + } + } + + #save() { + const altText = this.#textarea.value.trim(); + this.#currentEditor.altTextData = { + altText, + decorative: false, + hasAI: this.#hasAI, + }; + this.#telemetryData = { + action: "alt_text_save", + alt_text_description: !!altText, + alt_text_edit: + !!this.#previousAltText && this.#previousAltText !== altText, + alt_text_decorative: false, + alt_text_altered: + this.#guessedAltText && this.#guessedAltText !== altText, + }; + this.#finish(); + } + + destroy() { + this.#uiManager = null; // Avoid re-adding the edit listeners. + this.#finish(); + } +} + +export { NewAltTextManager }; diff --git a/web/stubs-geckoview.js b/web/stubs-geckoview.js index 726f03242d830a..3c0669d0d731d4 100644 --- a/web/stubs-geckoview.js +++ b/web/stubs-geckoview.js @@ -15,6 +15,7 @@ const AltTextManager = null; const AnnotationEditorParams = null; +const NewAltTextManager = null; const PDFAttachmentViewer = null; const PDFCursorTools = null; const PDFDocumentProperties = null; @@ -29,6 +30,7 @@ const SecondaryToolbar = null; export { AltTextManager, AnnotationEditorParams, + NewAltTextManager, PDFAttachmentViewer, PDFCursorTools, PDFDocumentProperties, diff --git a/web/viewer-geckoview.html b/web/viewer-geckoview.html index dee94a04d63940..6ddec5791b522b 100644 --- a/web/viewer-geckoview.html +++ b/web/viewer-geckoview.html @@ -65,6 +65,7 @@ "display-node_utils": "../src/display/stubs.js", "web-alt_text_manager": "./stubs-geckoview.js", + "web-new_alt_text_manager": "./stubs-geckoview.js", "web-annotation_editor_params": "./stubs-geckoview.js", "web-download_manager": "./download_manager.js", "web-external_services": "./genericcom.js", diff --git a/web/viewer.html b/web/viewer.html index 4175bcf5dbe80e..8920527cf94eb4 100644 --- a/web/viewer.html +++ b/web/viewer.html @@ -1,4 +1,4 @@ - +
diff --git a/web/viewer.js b/web/viewer.js index ff40485c37d7c4..3810d49c58382e 100644 --- a/web/viewer.js +++ b/web/viewer.js @@ -163,6 +163,32 @@ function getViewerConfiguration() { cancelButton: document.getElementById("altTextCancel"), saveButton: document.getElementById("altTextSave"), }, + newAltTextDialog: { + dialog: document.getElementById("newAltTextDialog"), + title: document.getElementById("newAltTextTitle"), + descriptionContainer: document.getElementById( + "newAltTextDescriptionContainer" + ), + textarea: document.getElementById("newAltTextDescriptionTextarea"), + disclaimer: document.getElementById("newAltTextDisclaimer"), + learnMore: document.getElementById("newAltTextLearnMore"), + imagePreview: document.getElementById("newAltTextImagePreview"), + createAutomatically: document.getElementById( + "newAltTextCreateAutomatically" + ), + createAutomaticallyButton: document.getElementById( + "newAltTextCreateAutomaticallyButton" + ), + downloadModel: document.getElementById("newAltTextDownloadModel"), + downloadModelDescription: document.getElementById( + "newAltTextDownloadModelDescription" + ), + error: document.getElementById("newAltTextError"), + errorCloseButton: document.getElementById("newAltTextCloseButton"), + cancelButton: document.getElementById("newAltTextCancel"), + notNowButton: document.getElementById("newAltTextNotNow"), + saveButton: document.getElementById("newAltTextSave"), + }, annotationEditorParams: { editorFreeTextFontSize: document.getElementById("editorFreeTextFontSize"), editorFreeTextColor: document.getElementById("editorFreeTextColor"),