diff --git a/.github/workflows/deploy-latest.yml b/.github/workflows/deploy-latest.yml index 5a5c6f504e2..4bf8d49bb64 100644 --- a/.github/workflows/deploy-latest.yml +++ b/.github/workflows/deploy-latest.yml @@ -8,7 +8,9 @@ permissions: pull-requests: write jobs: release-please: - runs-on: ubuntu-20.04 + runs-on: ubuntu-latest + permissions: + id-token: write steps: - uses: google-github-actions/release-please-action@v3.7.13 id: release diff --git a/.github/workflows/deploy-next.yml b/.github/workflows/deploy-next.yml index 906ea45c20e..7213f6a42a9 100644 --- a/.github/workflows/deploy-next.yml +++ b/.github/workflows/deploy-next.yml @@ -8,7 +8,9 @@ on: branches: [main] jobs: build: - runs-on: ubuntu-20.04 + runs-on: ubuntu-latest + permissions: + id-token: write steps: - uses: actions/checkout@v4 with: diff --git a/package-lock.json b/package-lock.json index 73277a31b42..6950d53964e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -36310,7 +36310,7 @@ "devDependencies": { "@esri/calcite-design-tokens": "^2.2.1-next.0", "@esri/calcite-ui-icons": "3.28.2", - "@esri/eslint-plugin-calcite-components": "^1.2.1-next.0", + "@esri/eslint-plugin-calcite-components": "^1.2.1-next.1", "@stencil-community/eslint-plugin": "0.7.2", "@stencil-community/postcss": "2.2.0", "@stencil/angular-output-target": "0.8.4", @@ -37893,7 +37893,7 @@ }, "packages/eslint-plugin-calcite-components": { "name": "@esri/eslint-plugin-calcite-components", - "version": "1.2.1-next.0", + "version": "1.2.1-next.1", "license": "SEE LICENSE.md", "dependencies": { "stencil-eslint-core": "0.4.1" diff --git a/packages/calcite-components-react/package.json b/packages/calcite-components-react/package.json index 8eda1e9ca61..eb209c4d16a 100644 --- a/packages/calcite-components-react/package.json +++ b/packages/calcite-components-react/package.json @@ -1,13 +1,24 @@ { "name": "@esri/calcite-components-react", - "sideEffects": false, "version": "2.9.0", - "homepage": "https://developers.arcgis.com/calcite-design-system/", "description": "A set of React components that wrap calcite components", + "homepage": "https://developers.arcgis.com/calcite-design-system/", + "repository": { + "type": "git", + "url": "git+https://github.com/Esri/calcite-design-system.git", + "directory": "packages/calcite-components-react" + }, "license": "SEE LICENSE.md", + "sideEffects": false, + "main": "./dist/index.js", + "module": "./dist/index.js", + "types": "./dist/index.d.ts", + "files": [ + "dist/" + ], "scripts": { - "build": "rimraf dist && npm run compile", "prebuild": "npm run patch:jsx-import", + "build": "rimraf dist && npm run compile", "clean": "rimraf dist node_modules .turbo", "compile": "npm run tsc", "lint": "concurrently npm:lint:*", @@ -16,12 +27,6 @@ "patch:jsx-import": "tsx support/patchJSXImport.ts", "tsc": "tsc" }, - "main": "./dist/index.js", - "module": "./dist/index.js", - "types": "./dist/index.d.ts", - "files": [ - "dist/" - ], "dependencies": { "@esri/calcite-components": "^2.9.0" }, diff --git a/packages/calcite-components/package.json b/packages/calcite-components/package.json index 85eb54a0576..57e4f0fd542 100644 --- a/packages/calcite-components/package.json +++ b/packages/calcite-components/package.json @@ -78,7 +78,7 @@ "devDependencies": { "@esri/calcite-design-tokens": "^2.2.1-next.0", "@esri/calcite-ui-icons": "3.28.2", - "@esri/eslint-plugin-calcite-components": "^1.2.1-next.0", + "@esri/eslint-plugin-calcite-components": "^1.2.1-next.1", "@stencil-community/eslint-plugin": "0.7.2", "@stencil-community/postcss": "2.2.0", "@stencil/angular-output-target": "0.8.4", diff --git a/packages/calcite-components/src/components/combobox/combobox.tsx b/packages/calcite-components/src/components/combobox/combobox.tsx index 32c09d60ceb..2aa5bc9bb56 100644 --- a/packages/calcite-components/src/components/combobox/combobox.tsx +++ b/packages/calcite-components/src/components/combobox/combobox.tsx @@ -350,7 +350,7 @@ export class Combobox await componentOnReady(this.el); - if (!this.allowCustomValues && this.textInput.value) { + if (!this.allowCustomValues && this.text) { this.clearInputValue(); this.filterItems(""); this.updateActiveItemIndex(-1); @@ -452,7 +452,7 @@ export class Combobox // // -------------------------------------------------------------------------- - connectedCallback(): void { + async connectedCallback(): Promise { connectInteractive(this); connectLocalized(this); connectMessages(this); @@ -472,7 +472,9 @@ export class Combobox onToggleOpenCloseComponent(this); } + await componentOnReady(this.el); connectFloatingUI(this, this.referenceEl, this.floatingEl); + afterConnectDefaultValueSet(this, this.getValue()); } async componentWillLoad(): Promise { @@ -482,8 +484,6 @@ export class Combobox } componentDidLoad(): void { - afterConnectDefaultValueSet(this, this.getValue()); - connectFloatingUI(this, this.referenceEl, this.floatingEl); setComponentLoaded(this); } diff --git a/packages/calcite-components/src/components/dropdown/dropdown.tsx b/packages/calcite-components/src/components/dropdown/dropdown.tsx index 5287df4b7b4..ad0a054680e 100644 --- a/packages/calcite-components/src/components/dropdown/dropdown.tsx +++ b/packages/calcite-components/src/components/dropdown/dropdown.tsx @@ -43,6 +43,7 @@ import { createObserver } from "../../utils/observers"; import { onToggleOpenCloseComponent, OpenCloseComponent } from "../../utils/openCloseComponent"; import { RequestedItem } from "../dropdown-group/interfaces"; import { Scale } from "../interfaces"; +import { componentOnReady } from "../../utils/component"; import { ItemKeyboardEvent } from "./interfaces"; import { SLOTS } from "./resources"; @@ -197,7 +198,7 @@ export class Dropdown // //-------------------------------------------------------------------------- - connectedCallback(): void { + async connectedCallback(): Promise { this.mutationObserver?.observe(this.el, { childList: true, subtree: true }); this.setFilteredPlacements(); if (this.open) { @@ -206,6 +207,8 @@ export class Dropdown } connectInteractive(this); this.updateItems(); + + await componentOnReady(this.el); connectFloatingUI(this, this.referenceEl, this.floatingEl); } @@ -215,7 +218,6 @@ export class Dropdown componentDidLoad(): void { setComponentLoaded(this); - connectFloatingUI(this, this.referenceEl, this.floatingEl); } componentDidRender(): void { diff --git a/packages/calcite-components/src/components/input-date-picker/input-date-picker.tsx b/packages/calcite-components/src/components/input-date-picker/input-date-picker.tsx index 7d8088c69eb..156ac9dd9c5 100644 --- a/packages/calcite-components/src/components/input-date-picker/input-date-picker.tsx +++ b/packages/calcite-components/src/components/input-date-picker/input-date-picker.tsx @@ -85,7 +85,7 @@ import { FocusTrapComponent, } from "../../utils/focusTrapComponent"; import { guid } from "../../utils/guid"; -import { getIconScale } from "../../utils/component"; +import { componentOnReady, getIconScale } from "../../utils/component"; import { Status } from "../interfaces"; import { Validation } from "../functional/Validation"; import { normalizeToCurrentCentury, isTwoDigitYear } from "./utils"; @@ -461,7 +461,7 @@ export class InputDatePicker // // -------------------------------------------------------------------------- - connectedCallback(): void { + async connectedCallback(): Promise { connectInteractive(this); connectLocalized(this); @@ -508,7 +508,9 @@ export class InputDatePicker onToggleOpenCloseComponent(this); } + await componentOnReady(this.el); connectFloatingUI(this, this.referenceEl, this.floatingEl); + this.localizeInputValues(); } async componentWillLoad(): Promise { @@ -520,8 +522,6 @@ export class InputDatePicker componentDidLoad(): void { setComponentLoaded(this); - this.localizeInputValues(); - connectFloatingUI(this, this.referenceEl, this.floatingEl); } disconnectedCallback(): void { diff --git a/packages/calcite-components/src/components/popover/popover.tsx b/packages/calcite-components/src/components/popover/popover.tsx index e0dd92c3c1c..c34fba1339a 100644 --- a/packages/calcite-components/src/components/popover/popover.tsx +++ b/packages/calcite-components/src/components/popover/popover.tsx @@ -55,7 +55,7 @@ import { } from "../../utils/loadable"; import { createObserver } from "../../utils/observers"; import { FloatingArrow } from "../functional/FloatingArrow"; -import { getIconScale } from "../../utils/component"; +import { componentOnReady, getIconScale } from "../../utils/component"; import PopoverManager from "./PopoverManager"; import { PopoverMessages } from "./assets/popover/t9n"; import { ARIA_CONTROLS, ARIA_EXPANDED, CSS, defaultPopoverPlacement } from "./resources"; @@ -278,8 +278,6 @@ export class Popover transitionEl: HTMLDivElement; - hasLoaded = false; - focusTrap: FocusTrap; // -------------------------------------------------------------------------- @@ -288,16 +286,18 @@ export class Popover // // -------------------------------------------------------------------------- - connectedCallback(): void { + async connectedCallback(): Promise { this.setFilteredPlacements(); connectLocalized(this); connectMessages(this); - this.setUpReferenceElement(this.hasLoaded); + + await componentOnReady(this.el); + this.setUpReferenceElement(); connectFocusTrap(this); + if (this.open) { onToggleOpenCloseComponent(this); } - connectFloatingUI(this, this.effectiveReferenceElement, this.el); } async componentWillLoad(): Promise { @@ -307,11 +307,6 @@ export class Popover componentDidLoad(): void { setComponentLoaded(this); - if (this.referenceElement && !this.effectiveReferenceElement) { - this.setUpReferenceElement(); - } - connectFloatingUI(this, this.effectiveReferenceElement, this.el); - this.hasLoaded = true; } disconnectedCallback(): void { diff --git a/packages/calcite-components/src/components/tooltip/tooltip.tsx b/packages/calcite-components/src/components/tooltip/tooltip.tsx index 13c843321a7..ebb760f83d2 100644 --- a/packages/calcite-components/src/components/tooltip/tooltip.tsx +++ b/packages/calcite-components/src/components/tooltip/tooltip.tsx @@ -27,6 +27,7 @@ import { import { guid } from "../../utils/guid"; import { onToggleOpenCloseComponent, OpenCloseComponent } from "../../utils/openCloseComponent"; import { FloatingArrow } from "../functional/FloatingArrow"; +import { componentOnReady } from "../../utils/component"; import { ARIA_DESCRIBED_BY, CSS } from "./resources"; import TooltipManager from "./TooltipManager"; import { getEffectiveReferenceElement } from "./utils"; @@ -146,8 +147,6 @@ export class Tooltip implements FloatingUIComponent, OpenCloseComponent { guid = `calcite-tooltip-${guid()}`; - hasLoaded = false; - openTransitionProp = "opacity"; transitionEl: HTMLDivElement; @@ -158,12 +157,13 @@ export class Tooltip implements FloatingUIComponent, OpenCloseComponent { // // -------------------------------------------------------------------------- - connectedCallback(): void { - this.setUpReferenceElement(this.hasLoaded); + async connectedCallback(): Promise { + await componentOnReady(this.el); + this.setUpReferenceElement(true); + if (this.open) { onToggleOpenCloseComponent(this); } - connectFloatingUI(this, this.effectiveReferenceElement, this.el); } async componentWillLoad(): Promise { @@ -172,14 +172,6 @@ export class Tooltip implements FloatingUIComponent, OpenCloseComponent { } } - componentDidLoad(): void { - if (this.referenceElement && !this.effectiveReferenceElement) { - this.setUpReferenceElement(); - } - connectFloatingUI(this, this.effectiveReferenceElement, this.el); - this.hasLoaded = true; - } - disconnectedCallback(): void { this.removeReferences(); disconnectFloatingUI(this, this.effectiveReferenceElement, this.el); diff --git a/packages/calcite-components/src/utils/floating-ui.spec.ts b/packages/calcite-components/src/utils/floating-ui.spec.ts index dd65a4bf420..6069b8f3350 100644 --- a/packages/calcite-components/src/utils/floating-ui.spec.ts +++ b/packages/calcite-components/src/utils/floating-ui.spec.ts @@ -3,7 +3,7 @@ import * as floatingUI from "./floating-ui"; import { FloatingUIComponent } from "./floating-ui"; const { - cleanupMap, + autoUpdatingComponentMap, connectFloatingUI, defaultOffsetDistance, disconnectFloatingUI, @@ -36,28 +36,40 @@ it("should set calcite placement to FloatingUI placement", () => { expect(getEffectivePlacement(el, "trailing-end")).toBe("left-end"); }); +function createFakeFloatingUiComponent(referenceEl: HTMLElement, floatingEl: HTMLElement): FloatingUIComponent { + const fake: FloatingUIComponent = { + open: false, + reposition: async () => { + await reposition(fake, { + floatingEl, + referenceEl, + overlayPositioning: fake.overlayPositioning, + placement: "top", + flipPlacements: [], + type: "menu", + }); + }, + overlayPositioning: "absolute", + placement: "auto", + }; + + return fake; +} + describe("repositioning", () => { let fakeFloatingUiComponent: FloatingUIComponent; let floatingEl: HTMLDivElement; let referenceEl: HTMLButtonElement; let positionOptions: Parameters[1]; - function createFakeFloatingUiComponent(): FloatingUIComponent { - return { - open: false, - reposition: async () => { - /* noop */ - }, - overlayPositioning: "absolute", - placement: "auto", - }; - } - beforeEach(() => { - fakeFloatingUiComponent = createFakeFloatingUiComponent(); - - floatingEl = document.createElement("div"); referenceEl = document.createElement("button"); + floatingEl = document.createElement("div"); + + document.body.append(floatingEl); + document.body.append(referenceEl); + + fakeFloatingUiComponent = createFakeFloatingUiComponent(referenceEl, floatingEl); positionOptions = { floatingEl, @@ -66,6 +78,8 @@ describe("repositioning", () => { placement: fakeFloatingUiComponent.placement, type: "popover", }; + + connectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); }); function assertPreOpenPositioning(floatingEl: HTMLElement): void { @@ -112,55 +126,70 @@ describe("repositioning", () => { assertOpenPositioning(floatingEl); }); - describe("connect/disconnect helpers", () => { - it("has connectedCallback and disconnectedCallback helpers", () => { - expect(cleanupMap.has(fakeFloatingUiComponent)).toBe(false); - expect(floatingEl.style.position).toBe(""); - expect(floatingEl.style.visibility).toBe(""); - expect(floatingEl.style.pointerEvents).toBe(""); + it("debounces positioning per instance", async () => { + const positionSpy = jest.spyOn(floatingUI, "positionFloatingUI"); + fakeFloatingUiComponent.open = true; - connectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); + const anotherFakeFloatingUiComponent = createFakeFloatingUiComponent(referenceEl, floatingEl); + anotherFakeFloatingUiComponent.open = true; - expect(cleanupMap.has(fakeFloatingUiComponent)).toBe(true); - expect(floatingEl.style.position).toBe("absolute"); - expect(floatingEl.style.visibility).toBe("hidden"); - expect(floatingEl.style.pointerEvents).toBe("none"); + floatingUI.reposition(fakeFloatingUiComponent, positionOptions, true); + expect(positionSpy).toHaveBeenCalledTimes(1); - disconnectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); + floatingUI.reposition(anotherFakeFloatingUiComponent, positionOptions, true); + expect(positionSpy).toHaveBeenCalledTimes(2); - expect(cleanupMap.has(fakeFloatingUiComponent)).toBe(false); - expect(floatingEl.style.position).toBe("absolute"); + await new Promise((resolve) => setTimeout(resolve, repositionDebounceTimeout)); + expect(positionSpy).toHaveBeenCalledTimes(2); + }); +}); - fakeFloatingUiComponent.overlayPositioning = "fixed"; - connectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); +describe("connect/disconnect helpers", () => { + let fakeFloatingUiComponent: FloatingUIComponent; + let floatingEl: HTMLDivElement; + let referenceEl: HTMLButtonElement; - expect(cleanupMap.has(fakeFloatingUiComponent)).toBe(true); - expect(floatingEl.style.position).toBe("fixed"); - expect(floatingEl.style.visibility).toBe("hidden"); - expect(floatingEl.style.pointerEvents).toBe("none"); + beforeEach(() => { + referenceEl = document.createElement("button"); + floatingEl = document.createElement("div"); - disconnectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); + document.body.append(floatingEl); + document.body.append(referenceEl); - expect(cleanupMap.has(fakeFloatingUiComponent)).toBe(false); - expect(floatingEl.style.position).toBe("fixed"); - }); + fakeFloatingUiComponent = createFakeFloatingUiComponent(referenceEl, floatingEl); }); - it("debounces positioning per instance", async () => { - const positionSpy = jest.spyOn(floatingUI, "positionFloatingUI"); + it("has connectedCallback and disconnectedCallback helpers", async () => { fakeFloatingUiComponent.open = true; + expect(autoUpdatingComponentMap.has(fakeFloatingUiComponent)).toBe(false); + expect(floatingEl.style.position).toBe(""); + expect(floatingEl.style.visibility).toBe(""); + expect(floatingEl.style.pointerEvents).toBe(""); - const anotherFakeFloatingUiComponent = createFakeFloatingUiComponent(); - anotherFakeFloatingUiComponent.open = true; + await connectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); - floatingUI.reposition(fakeFloatingUiComponent, positionOptions, true); - expect(positionSpy).toHaveBeenCalledTimes(1); + expect(autoUpdatingComponentMap.has(fakeFloatingUiComponent)).toBe(true); + expect(floatingEl.style.position).toBe("absolute"); + expect(floatingEl.style.visibility).toBe("hidden"); + expect(floatingEl.style.pointerEvents).toBe("none"); - floatingUI.reposition(anotherFakeFloatingUiComponent, positionOptions, true); - expect(positionSpy).toHaveBeenCalledTimes(2); + disconnectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); - await new Promise((resolve) => setTimeout(resolve, repositionDebounceTimeout)); - expect(positionSpy).toHaveBeenCalledTimes(2); + expect(autoUpdatingComponentMap.has(fakeFloatingUiComponent)).toBe(false); + expect(floatingEl.style.position).toBe("absolute"); + + fakeFloatingUiComponent.overlayPositioning = "fixed"; + await connectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); + + expect(autoUpdatingComponentMap.has(fakeFloatingUiComponent)).toBe(true); + expect(floatingEl.style.position).toBe("fixed"); + expect(floatingEl.style.visibility).toBe("hidden"); + expect(floatingEl.style.pointerEvents).toBe("none"); + + disconnectFloatingUI(fakeFloatingUiComponent, referenceEl, floatingEl); + + expect(autoUpdatingComponentMap.has(fakeFloatingUiComponent)).toBe(false); + expect(floatingEl.style.position).toBe("fixed"); }); }); diff --git a/packages/calcite-components/src/utils/floating-ui.ts b/packages/calcite-components/src/utils/floating-ui.ts index 2687c46e999..9d921f4f100 100644 --- a/packages/calcite-components/src/utils/floating-ui.ts +++ b/packages/calcite-components/src/utils/floating-ui.ts @@ -427,13 +427,19 @@ export async function reposition( options: Parameters[1], delayed = false, ): Promise { - if (!component.open) { + if (!component.open || !options.floatingEl || !options.referenceEl) { return; } + const trackedState = autoUpdatingComponentMap.get(component); + + if (!trackedState) { + return runAutoUpdate(component, options.referenceEl, options.floatingEl); + } + const positionFunction = delayed ? getDebouncedReposition(component) : positionFloatingUI; - return positionFunction(component, options); + await positionFunction(component, options); } function getDebouncedReposition(component: FloatingUIComponent): DebouncedFunc { @@ -460,15 +466,67 @@ const ARROW_CSS_TRANSFORM = { right: "rotate(90deg)", }; +type PendingFloatingUIState = { + state: "pending"; +}; + +type ActiveFloatingUIState = { + state: "active"; + cleanUp: () => void; +}; + +type TrackedFloatingUIState = PendingFloatingUIState | ActiveFloatingUIState; + /** * Exported for testing purposes only * * @internal */ -export const cleanupMap = new WeakMap void>(); +export const autoUpdatingComponentMap = new WeakMap(); const componentToDebouncedRepositionMap = new WeakMap>(); +async function runAutoUpdate( + component: FloatingUIComponent, + referenceEl: ReferenceElement, + floatingEl: HTMLElement, +): Promise { + if (!floatingEl.isConnected) { + return; + } + + const effectiveAutoUpdate = Build.isBrowser + ? autoUpdate + : (_refEl: HTMLElement, _floatingEl: HTMLElement, updateCallback: () => void): (() => void) => { + updateCallback(); + return () => { + /* noop */ + }; + }; + + // we set initial state here to make it available for `reposition` calls + autoUpdatingComponentMap.set(component, { state: "pending" }); + + let repositionPromise: Promise; + + const cleanUp = effectiveAutoUpdate( + referenceEl, + floatingEl, + // callback is invoked immediately + () => { + const promise = component.reposition(); + + if (!repositionPromise) { + repositionPromise = promise; + } + }, + ); + + autoUpdatingComponentMap.set(component, { state: "active", cleanUp }); + + return repositionPromise; +} + /** * Helper to set up floating element interactions on connectedCallback. * @@ -476,11 +534,11 @@ const componentToDebouncedRepositionMap = new WeakMap { if (!floatingEl || !referenceEl) { return; } @@ -495,19 +553,11 @@ export function connectFloatingUI( position: component.overlayPositioning, }); - const runAutoUpdate = Build.isBrowser - ? autoUpdate - : (_refEl: HTMLElement, _floatingEl: HTMLElement, updateCallback: () => void): (() => void) => { - updateCallback(); - return () => { - /* noop */ - }; - }; + if (!component.open) { + return; + } - cleanupMap.set( - component, - runAutoUpdate(referenceEl, floatingEl, () => component.reposition()), - ); + return runAutoUpdate(component, referenceEl, floatingEl); } /** @@ -526,8 +576,13 @@ export function disconnectFloatingUI( return; } - cleanupMap.get(component)?.(); - cleanupMap.delete(component); + const trackedState = autoUpdatingComponentMap.get(component); + + if (trackedState?.state === "active") { + trackedState.cleanUp(); + } + + autoUpdatingComponentMap.delete(component); componentToDebouncedRepositionMap.get(component)?.cancel(); componentToDebouncedRepositionMap.delete(component); diff --git a/packages/eslint-plugin-calcite-components/package.json b/packages/eslint-plugin-calcite-components/package.json index 6700f3de1e2..a60e50f40b9 100644 --- a/packages/eslint-plugin-calcite-components/package.json +++ b/packages/eslint-plugin-calcite-components/package.json @@ -1,7 +1,13 @@ { "name": "@esri/eslint-plugin-calcite-components", - "version": "1.2.1-next.0", + "version": "1.2.1-next.1", "description": "ESLint rules for @esri/calcite-components", + "repository": { + "type": "git", + "url": "git+https://github.com/Esri/calcite-design-system.git", + "directory": "packages/eslint-plugin-calcite-components" + }, + "license": "SEE LICENSE.md", "main": "dist/index.js", "files": [ "dist/index.js" @@ -16,15 +22,14 @@ "dependencies": { "stencil-eslint-core": "0.4.1" }, + "devDependencies": { + "ts-node": "10.9.2" + }, "peerDependencies": { "eslint": ">=8.0.0" }, - "license": "SEE LICENSE.md", "packageManager": "npm@8.19.4", "volta": { "extends": "../../package.json" - }, - "devDependencies": { - "ts-node": "10.9.2" } }