diff --git a/frontend/package.json b/frontend/package.json
index ab4e391c1ee..7764028724e 100644
--- a/frontend/package.json
+++ b/frontend/package.json
@@ -170,6 +170,7 @@
"use-resize-observer": "^9.1.0",
"vega-lite": "^5.23.0",
"vega-loader": "^4.5.3",
+ "vega-tooltip": "^1.1.0",
"vscode-jsonrpc": "^8.2.1",
"vscode-languageserver-protocol": "^3.17.5",
"web-vitals": "^4.2.4",
diff --git a/frontend/src/components/charts/tooltip.ts b/frontend/src/components/charts/tooltip.ts
new file mode 100644
index 00000000000..ca1ea6621a1
--- /dev/null
+++ b/frontend/src/components/charts/tooltip.ts
@@ -0,0 +1,4 @@
+import { Handler } from "vega-tooltip";
+
+// Create a tooltip handler that supports HTML content (including images)
+export const tooltipHandler = new Handler();
diff --git a/frontend/src/components/data-table/charts/lazy-chart.tsx b/frontend/src/components/data-table/charts/lazy-chart.tsx
index 305f3d6f9d2..c1c8235e15e 100644
--- a/frontend/src/components/data-table/charts/lazy-chart.tsx
+++ b/frontend/src/components/data-table/charts/lazy-chart.tsx
@@ -2,6 +2,7 @@
import React from "react";
import type { TopLevelSpec } from "vega-lite";
+import { tooltipHandler } from "@/components/charts/tooltip";
import { useTheme } from "@/theme/useTheme";
import type { ErrorMessage } from "./chart-spec/spec";
import { augmentSpecWithData } from "./chart-spec/spec";
@@ -33,6 +34,7 @@ export const LazyChart: React.FC<{
spec={spec}
theme={theme === "dark" ? "dark" : undefined}
height={height}
+ tooltip={tooltipHandler.call}
actions={{
export: true,
source: false,
diff --git a/frontend/src/components/editor/Output.tsx b/frontend/src/components/editor/Output.tsx
index 30464fc7eb3..dd6639e8ab7 100644
--- a/frontend/src/components/editor/Output.tsx
+++ b/frontend/src/components/editor/Output.tsx
@@ -25,6 +25,7 @@ import {
ChevronsUpDownIcon,
ExpandIcon,
} from "lucide-react";
+import { tooltipHandler } from "@/components/charts/tooltip";
import { useExpandedOutput } from "@/core/cells/outputs";
import { useIframeCapabilities } from "@/hooks/useIframeCapabilities";
import { renderHTML } from "@/plugins/core/RenderHTML";
@@ -179,6 +180,7 @@ export const OutputRenderer: React.FC<{
}>
diff --git a/frontend/src/plugins/impl/data-explorer/ConnectedDataExplorerComponent.tsx b/frontend/src/plugins/impl/data-explorer/ConnectedDataExplorerComponent.tsx
index 613d1fd3951..279d6c0e1bd 100644
--- a/frontend/src/plugins/impl/data-explorer/ConnectedDataExplorerComponent.tsx
+++ b/frontend/src/plugins/impl/data-explorer/ConnectedDataExplorerComponent.tsx
@@ -6,6 +6,7 @@ import { ListFilterIcon } from "lucide-react";
import React, { type JSX, useMemo } from "react";
import { VegaLite } from "react-vega";
import type { VegaLiteProps } from "react-vega/lib/VegaLite";
+import { tooltipHandler } from "@/components/charts/tooltip";
import { Badge } from "@/components/ui/badge";
import { Button } from "@/components/ui/button";
import { Tooltip } from "@/components/ui/tooltip";
@@ -135,6 +136,7 @@ export const DataExplorerComponent = ({
padding={PADDING}
actions={ACTIONS}
spec={makeResponsive(mainPlot.spec)}
+ tooltip={tooltipHandler.call}
theme={theme === "dark" ? "dark" : undefined}
/>
@@ -210,6 +212,7 @@ export const DataExplorerComponent = ({
key={idx}
actions={false}
spec={plot.spec}
+ tooltip={tooltipHandler.call}
theme={theme === "dark" ? "dark" : undefined}
/>
diff --git a/frontend/src/plugins/impl/vega/vega-component.tsx b/frontend/src/plugins/impl/vega/vega-component.tsx
index 1612946802b..61aca665a57 100644
--- a/frontend/src/plugins/impl/vega/vega-component.tsx
+++ b/frontend/src/plugins/impl/vega/vega-component.tsx
@@ -8,6 +8,7 @@ import useEvent from "react-use-event-hook";
import { type SignalListeners, VegaLite, type View } from "react-vega";
// @ts-expect-error vega-typings does not include formats
import { formats } from "vega-loader";
+import { tooltipHandler } from "@/components/charts/tooltip";
import { Alert, AlertTitle } from "@/components/ui/alert";
import { Tooltip } from "@/components/ui/tooltip";
import { useAsyncData } from "@/hooks/useAsyncData";
@@ -26,7 +27,6 @@ import type { VegaLiteSpec } from "./types";
// register arrow reader under type 'arrow'
formats("arrow", arrow);
-
export interface Data {
spec: VegaLiteSpec;
chartSelection: boolean | "point" | "interval";
@@ -234,6 +234,7 @@ const LoadedVegaComponent = ({
signalListeners={signalListeners}
onError={handleError}
onNewView={handleNewView}
+ tooltip={tooltipHandler.call}
/>
{renderHelpContent()}
diff --git a/marimo/_smoke_tests/altair_examples/tooltip_images.py b/marimo/_smoke_tests/altair_examples/tooltip_images.py
new file mode 100644
index 00000000000..f4b06cd3624
--- /dev/null
+++ b/marimo/_smoke_tests/altair_examples/tooltip_images.py
@@ -0,0 +1,206 @@
+# /// script
+# requires-python = ">=3.11"
+# dependencies = [
+# "pandas",
+# "altair",
+# "marimo",
+# ]
+# ///
+# Copyright 2024 Marimo. All rights reserved.
+
+import marimo
+
+__generated_with = "0.17.2"
+app = marimo.App(width="medium")
+
+
+@app.cell
+def _():
+ import marimo as mo
+ return (mo,)
+
+
+@app.cell(hide_code=True)
+def _(mo):
+ mo.md(
+ """
+ # Altair Tooltip Images
+
+ This smoke test verifies that images in Altair tooltips render correctly
+ and that the tooltip handler properly sanitizes potentially malicious content.
+
+ **Tests:**
+ 1. External image URLs in tooltips
+ 2. Base64-encoded images in tooltips
+ 3. XSS vulnerability prevention
+ """
+ )
+ return
+
+
+@app.cell
+def _():
+ import altair as alt
+ import pandas as pd
+
+ # Create sample data with image URLs
+ source = pd.DataFrame(
+ {
+ "a": [1, 2],
+ "b": [1, 2],
+ "image": [
+ "https://marimo.io/logo.png",
+ "https://marimo.io/favicon.ico",
+ ],
+ }
+ )
+
+ # Create chart with image tooltip
+ chart = (
+ alt.Chart(source)
+ .mark_circle(size=200)
+ .encode(
+ x=alt.X("a", scale=alt.Scale(domain=[0, 3])),
+ y=alt.Y("b", scale=alt.Scale(domain=[0, 3])),
+ tooltip=["image"],
+ )
+ .properties(
+ title="Scatter Plot with Image Tooltips - Hover to see images",
+ width=400,
+ height=400,
+ )
+ )
+
+ chart
+ return alt, pd
+
+
+@app.cell(hide_code=True)
+def _(mo):
+ mo.md(
+ r"""
+ ## Instructions
+
+ Hover over the circles to see the image tooltips.
+ The images should render in the tooltip, not just show URLs as text.
+ """
+ )
+ return
+
+
+@app.cell
+def _(alt, pd):
+ # Example with base64 encoded image
+ base64_image = ""
+
+ source2 = pd.DataFrame(
+ {
+ "x": [1, 2, 3],
+ "y": [1, 2, 1],
+ "name": ["Red Circle", "Blue Square", "Green Triangle"],
+ "image": [
+ base64_image,
+ "",
+ "",
+ ],
+ }
+ )
+
+ chart2 = (
+ alt.Chart(source2)
+ .mark_point(size=300, filled=True)
+ .encode(
+ x=alt.X("x", scale=alt.Scale(domain=[0, 4])),
+ y=alt.Y("y", scale=alt.Scale(domain=[0, 3])),
+ tooltip=["name", "image"],
+ )
+ .properties(
+ title="Chart with Base64 Encoded Image Tooltips",
+ width=400,
+ height=300,
+ )
+ )
+
+ chart2
+ return
+
+
+@app.cell(hide_code=True)
+def _(mo):
+ mo.md(
+ r"""
+ ## XSS Security Tests
+
+ These tests verify that the tooltip handler properly sanitizes
+ potentially malicious content and prevents XSS attacks.
+
+ **Expected behavior:** All XSS attempts should be neutralized.
+ No alert boxes or script execution should occur when hovering.
+ """
+ )
+ return
+
+
+@app.cell
+def _(alt, pd):
+ # Test various XSS attack vectors
+ xss_test_data = pd.DataFrame(
+ {
+ "x": [1, 2, 3, 4, 1, 2, 3, 4],
+ "y": [1, 1, 1, 1, 2, 2, 2, 2],
+ "label": [
+ "Script Tag",
+ "Event Handler",
+ "JS URL",
+ "IMG onerror",
+ "SVG Script",
+ "HTML Injection",
+ "OnMouseOver",
+ "Data URL JS",
+ ],
+ "image": [
+ # Script tag injection
+ '',
+ # Event handler
+ "
",
+ # JavaScript URL
+ "click",
+ # IMG with onerror
+ '
',
+ # SVG with embedded script
+ '',
+ # HTML injection
+ "
Click me
",
+ # OnMouseOver
+ "hover",
+ # Data URL with JavaScript
+ "
alert('XSS')\">",
+ ],
+ }
+ )
+
+ xss_chart = (
+ alt.Chart(xss_test_data)
+ .mark_circle(size=150)
+ .encode(
+ x=alt.X("x:O", title="Test Vector"),
+ y=alt.Y("y:O", title="Category"),
+ color=alt.Color(
+ "label:N",
+ legend=alt.Legend(title="Attack Type"),
+ ),
+ tooltip=["label", "image"],
+ )
+ .properties(
+ title="XSS Security Test - Hover to verify sanitization",
+ width=600,
+ height=300,
+ )
+ )
+
+ xss_chart
+ return
+
+
+if __name__ == "__main__":
+ app.run()
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index 086b7309739..a4ae5beaf2e 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -496,6 +496,9 @@ importers:
vega-loader:
specifier: ^4.5.3
version: 4.5.3
+ vega-tooltip:
+ specifier: ^1.1.0
+ version: 1.1.0
vscode-jsonrpc:
specifier: ^8.2.1
version: 8.2.1
@@ -9813,6 +9816,9 @@ packages:
vega-tooltip@0.22.1:
resolution: {integrity: sha512-mPmzxwvi6+2ZgbZ/+mNC7XbSu5I6Ckon8zdgUfH9neb+vV7CKlV/FYypMdVN/9iDMFUqGzybYdqNOiSPPIxFEQ==}
+ vega-tooltip@1.1.0:
+ resolution: {integrity: sha512-PP4CxC8gX//SBUtlcJkwffmvdZBvzAsqS0EANBKvImJ9PxV/KtJkcs7RCqp+A7nh2cjWdVzyOBWAvqKhXJStTQ==}
+
vega-transforms@4.12.1:
resolution: {integrity: sha512-Qxo+xeEEftY1jYyKgzOGc9NuW4/MqGm1YPZ5WrL9eXg2G0410Ne+xL/MFIjHF4hRX+3mgFF4Io2hPpfy/thjLg==}
@@ -9828,6 +9834,9 @@ packages:
vega-util@1.17.3:
resolution: {integrity: sha512-nSNpZLUrRvFo46M5OK4O6x6f08WD1yOcEzHNlqivF+sDLSsVpstaF6fdJYwrbf/debFi2L9Tkp4gZQtssup9iQ==}
+ vega-util@2.1.0:
+ resolution: {integrity: sha512-PGfp0m0QCufDmcxKJCWQy4Ov23FoF8DSXmoJwSezi3itQaa2hbxK0+xwsTMP2vy4PR16Pu25HMzgMwXVW1+33w==}
+
vega-view-transforms@4.6.1:
resolution: {integrity: sha512-RYlyMJu5kZV4XXjmyTQKADJWDB25SMHsiF+B1rbE1p+pmdQPlp5tGdPl9r5dUJOp3p8mSt/NGI8GPGucmPMxtw==}
@@ -21483,6 +21492,10 @@ snapshots:
dependencies:
vega-util: 1.17.3
+ vega-tooltip@1.1.0:
+ dependencies:
+ vega-util: 2.1.0
+
vega-transforms@4.12.1:
dependencies:
d3-array: 3.2.4
@@ -21508,6 +21521,8 @@ snapshots:
vega-util@1.17.3: {}
+ vega-util@2.1.0: {}
+
vega-view-transforms@4.6.1:
dependencies:
vega-dataflow: 5.7.7