{filters.map((filter, i) => (
{
+ window.innerHeight = 500;
+});
+
+afterEach(() => {
+ window.innerHeight = originalInnerHeight;
+});
+
+test('correctly positions at upper edge of screen', () => {
+ expect(getMenuAdjustedY(75, 1)).toEqual(75); // No adjustment
+ expect(getMenuAdjustedY(75, 2)).toEqual(75); // No adjustment
+ expect(getMenuAdjustedY(75, 3)).toEqual(75); // No adjustment
+});
+
+test('correctly positions at lower edge of screen', () => {
+ expect(getMenuAdjustedY(425, 1)).toEqual(425); // No adjustment
+ expect(getMenuAdjustedY(425, 2)).toEqual(404); // Adjustment
+ expect(getMenuAdjustedY(425, 3)).toEqual(372); // Adjustment
+});
diff --git a/superset-frontend/src/components/Chart/utils.ts b/superset-frontend/src/components/Chart/utils.ts
new file mode 100644
index 000000000..54fc5e892
--- /dev/null
+++ b/superset-frontend/src/components/Chart/utils.ts
@@ -0,0 +1,40 @@
+/**
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements. See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership. The ASF licenses this file
+ * to you 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.
+ */
+
+export const MENU_ITEM_HEIGHT = 32;
+const MENU_VERTICAL_SPACING = 32;
+
+/**
+ * Calculates an adjusted Y-offset for a menu or submenu to prevent that
+ * menu from appearing offscreen
+ *
+ * @param clientY The original Y-offset
+ * @param itemsCount The number of menu items
+ */
+export function getMenuAdjustedY(clientY: number, itemsCount: number) {
+ // Viewport height
+ const vh = Math.max(
+ document.documentElement.clientHeight || 0,
+ window.innerHeight || 0,
+ );
+
+ const menuHeight = MENU_ITEM_HEIGHT * itemsCount + MENU_VERTICAL_SPACING;
+ // Always show the context menu inside the viewport
+ return vh - clientY < menuHeight ? vh - menuHeight : clientY;
+}