Files
hive/frontend/node_modules/@headlessui/react/dist/headlessui.dev.cjs
anthonyrawlins 85bf1341f3 Add comprehensive frontend UI and distributed infrastructure
Frontend Enhancements:
- Complete React TypeScript frontend with modern UI components
- Distributed workflows management interface with real-time updates
- Socket.IO integration for live agent status monitoring
- Agent management dashboard with cluster visualization
- Project management interface with metrics and task tracking
- Responsive design with proper error handling and loading states

Backend Infrastructure:
- Distributed coordinator for multi-agent workflow orchestration
- Cluster management API with comprehensive agent operations
- Enhanced database models for agents and projects
- Project service for filesystem-based project discovery
- Performance monitoring and metrics collection
- Comprehensive API documentation and error handling

Documentation:
- Complete distributed development guide (README_DISTRIBUTED.md)
- Comprehensive development report with architecture insights
- System configuration templates and deployment guides

The platform now provides a complete web interface for managing the distributed AI cluster
with real-time monitoring, workflow orchestration, and agent coordination capabilities.

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-07-10 08:41:59 +10:00

8419 lines
272 KiB
JavaScript

"use strict";
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
// If the importer is in node compatibility mode or this is not an ESM
// file that has been converted to a CommonJS file using a Babel-
// compatible transform (i.e. "__esModule" has not been set), then set
// "default" to the CommonJS "module.exports" for node compatibility.
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
mod
));
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var __publicField = (obj, key, value) => {
__defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
return value;
};
// src/index.ts
var src_exports = {};
__export(src_exports, {
Combobox: () => Combobox,
Dialog: () => Dialog,
Disclosure: () => Disclosure,
FocusTrap: () => FocusTrap,
Listbox: () => Listbox,
Menu: () => Menu,
Popover: () => Popover,
Portal: () => Portal,
RadioGroup: () => RadioGroup,
Switch: () => Switch,
Tab: () => Tab,
Transition: () => Transition
});
module.exports = __toCommonJS(src_exports);
// ../../node_modules/@tanstack/react-virtual/build/lib/_virtual/_rollupPluginBabelHelpers.mjs
function _extends() {
_extends = Object.assign ? Object.assign.bind() : function(target) {
for (var i = 1; i < arguments.length; i++) {
var source = arguments[i];
for (var key in source) {
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
return _extends.apply(this, arguments);
}
// ../../node_modules/@tanstack/react-virtual/build/lib/index.mjs
var React = __toESM(require("react"), 1);
// ../../node_modules/@tanstack/virtual-core/build/lib/_virtual/_rollupPluginBabelHelpers.mjs
function _extends2() {
_extends2 = Object.assign ? Object.assign.bind() : function(target) {
for (var i = 1; i < arguments.length; i++) {
var source = arguments[i];
for (var key in source) {
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
return _extends2.apply(this, arguments);
}
// ../../node_modules/@tanstack/virtual-core/build/lib/utils.mjs
function memo(getDeps, fn, opts) {
var _opts$initialDeps;
var deps = (_opts$initialDeps = opts.initialDeps) != null ? _opts$initialDeps : [];
var result;
return function() {
var depTime;
if (opts.key && opts.debug != null && opts.debug())
depTime = Date.now();
var newDeps = getDeps();
var depsChanged = newDeps.length !== deps.length || newDeps.some(function(dep, index) {
return deps[index] !== dep;
});
if (!depsChanged) {
return result;
}
deps = newDeps;
var resultTime;
if (opts.key && opts.debug != null && opts.debug())
resultTime = Date.now();
result = fn.apply(void 0, newDeps);
if (opts.key && opts.debug != null && opts.debug()) {
var depEndTime = Math.round((Date.now() - depTime) * 100) / 100;
var resultEndTime = Math.round((Date.now() - resultTime) * 100) / 100;
var resultFpsPercentage = resultEndTime / 16;
var pad = function pad2(str, num) {
str = String(str);
while (str.length < num) {
str = " " + str;
}
return str;
};
console.info("%c\u23F1 " + pad(resultEndTime, 5) + " /" + pad(depEndTime, 5) + " ms", "\n font-size: .6rem;\n font-weight: bold;\n color: hsl(" + Math.max(0, Math.min(120 - 120 * resultFpsPercentage, 120)) + "deg 100% 31%);", opts == null ? void 0 : opts.key);
}
opts == null ? void 0 : opts.onChange == null ? void 0 : opts.onChange(result);
return result;
};
}
function notUndefined(value, msg) {
if (value === void 0) {
throw new Error("Unexpected undefined" + (msg ? ": " + msg : ""));
} else {
return value;
}
}
var approxEqual = function approxEqual2(a, b) {
return Math.abs(a - b) < 1;
};
// ../../node_modules/@tanstack/virtual-core/build/lib/index.mjs
var defaultKeyExtractor = function defaultKeyExtractor2(index) {
return index;
};
var defaultRangeExtractor = function defaultRangeExtractor2(range) {
var start = Math.max(range.startIndex - range.overscan, 0);
var end = Math.min(range.endIndex + range.overscan, range.count - 1);
var arr = [];
for (var _i = start; _i <= end; _i++) {
arr.push(_i);
}
return arr;
};
var observeElementRect = function observeElementRect2(instance, cb) {
var element = instance.scrollElement;
if (!element) {
return;
}
var handler = function handler2(rect) {
var width = rect.width, height = rect.height;
cb({
width: Math.round(width),
height: Math.round(height)
});
};
handler(element.getBoundingClientRect());
var observer = new ResizeObserver(function(entries) {
var entry = entries[0];
if (entry != null && entry.borderBoxSize) {
var box = entry.borderBoxSize[0];
if (box) {
handler({
width: box.inlineSize,
height: box.blockSize
});
return;
}
}
handler(element.getBoundingClientRect());
});
observer.observe(element, {
box: "border-box"
});
return function() {
observer.unobserve(element);
};
};
var observeElementOffset = function observeElementOffset2(instance, cb) {
var element = instance.scrollElement;
if (!element) {
return;
}
var handler = function handler2() {
cb(element[instance.options.horizontal ? "scrollLeft" : "scrollTop"]);
};
handler();
element.addEventListener("scroll", handler, {
passive: true
});
return function() {
element.removeEventListener("scroll", handler);
};
};
var measureElement = function measureElement2(element, entry, instance) {
if (entry != null && entry.borderBoxSize) {
var box = entry.borderBoxSize[0];
if (box) {
var size = Math.round(box[instance.options.horizontal ? "inlineSize" : "blockSize"]);
return size;
}
}
return Math.round(element.getBoundingClientRect()[instance.options.horizontal ? "width" : "height"]);
};
var elementScroll = function elementScroll2(offset, _ref2, instance) {
var _instance$scrollEleme3, _instance$scrollEleme4;
var _ref2$adjustments = _ref2.adjustments, adjustments = _ref2$adjustments === void 0 ? 0 : _ref2$adjustments, behavior = _ref2.behavior;
var toOffset = offset + adjustments;
(_instance$scrollEleme3 = instance.scrollElement) == null ? void 0 : _instance$scrollEleme3.scrollTo == null ? void 0 : _instance$scrollEleme3.scrollTo((_instance$scrollEleme4 = {}, _instance$scrollEleme4[instance.options.horizontal ? "left" : "top"] = toOffset, _instance$scrollEleme4.behavior = behavior, _instance$scrollEleme4));
};
var Virtualizer = function Virtualizer2(_opts) {
var _this = this;
this.unsubs = [];
this.scrollElement = null;
this.isScrolling = false;
this.isScrollingTimeoutId = null;
this.scrollToIndexTimeoutId = null;
this.measurementsCache = [];
this.itemSizeCache = /* @__PURE__ */ new Map();
this.pendingMeasuredCacheIndexes = [];
this.scrollDirection = null;
this.scrollAdjustments = 0;
this.measureElementCache = /* @__PURE__ */ new Map();
this.observer = function() {
var _ro = null;
var get = function get2() {
if (_ro) {
return _ro;
} else if (typeof ResizeObserver !== "undefined") {
return _ro = new ResizeObserver(function(entries) {
entries.forEach(function(entry) {
_this._measureElement(entry.target, entry);
});
});
} else {
return null;
}
};
return {
disconnect: function disconnect() {
var _get;
return (_get = get()) == null ? void 0 : _get.disconnect();
},
observe: function observe(target) {
var _get2;
return (_get2 = get()) == null ? void 0 : _get2.observe(target, {
box: "border-box"
});
},
unobserve: function unobserve(target) {
var _get3;
return (_get3 = get()) == null ? void 0 : _get3.unobserve(target);
}
};
}();
this.range = {
startIndex: 0,
endIndex: 0
};
this.setOptions = function(opts) {
Object.entries(opts).forEach(function(_ref3) {
var key = _ref3[0], value = _ref3[1];
if (typeof value === "undefined")
delete opts[key];
});
_this.options = _extends2({
debug: false,
initialOffset: 0,
overscan: 1,
paddingStart: 0,
paddingEnd: 0,
scrollPaddingStart: 0,
scrollPaddingEnd: 0,
horizontal: false,
getItemKey: defaultKeyExtractor,
rangeExtractor: defaultRangeExtractor,
onChange: function onChange() {
},
measureElement,
initialRect: {
width: 0,
height: 0
},
scrollMargin: 0,
scrollingDelay: 150,
indexAttribute: "data-index",
initialMeasurementsCache: [],
lanes: 1
}, opts);
};
this.notify = function() {
_this.options.onChange == null ? void 0 : _this.options.onChange(_this);
};
this.cleanup = function() {
_this.unsubs.filter(Boolean).forEach(function(d) {
return d();
});
_this.unsubs = [];
_this.scrollElement = null;
};
this._didMount = function() {
_this.measureElementCache.forEach(_this.observer.observe);
return function() {
_this.observer.disconnect();
_this.cleanup();
};
};
this._willUpdate = function() {
var scrollElement = _this.options.getScrollElement();
if (_this.scrollElement !== scrollElement) {
_this.cleanup();
_this.scrollElement = scrollElement;
_this._scrollToOffset(_this.scrollOffset, {
adjustments: void 0,
behavior: void 0
});
_this.unsubs.push(_this.options.observeElementRect(_this, function(rect) {
var prev = _this.scrollRect;
_this.scrollRect = rect;
if (_this.options.horizontal ? rect.width !== prev.width : rect.height !== prev.height) {
_this.maybeNotify();
}
}));
_this.unsubs.push(_this.options.observeElementOffset(_this, function(offset) {
_this.scrollAdjustments = 0;
if (_this.scrollOffset === offset) {
return;
}
if (_this.isScrollingTimeoutId !== null) {
clearTimeout(_this.isScrollingTimeoutId);
_this.isScrollingTimeoutId = null;
}
_this.isScrolling = true;
_this.scrollDirection = _this.scrollOffset < offset ? "forward" : "backward";
_this.scrollOffset = offset;
_this.maybeNotify();
_this.isScrollingTimeoutId = setTimeout(function() {
_this.isScrollingTimeoutId = null;
_this.isScrolling = false;
_this.scrollDirection = null;
_this.maybeNotify();
}, _this.options.scrollingDelay);
}));
}
};
this.getSize = function() {
return _this.scrollRect[_this.options.horizontal ? "width" : "height"];
};
this.memoOptions = memo(function() {
return [_this.options.count, _this.options.paddingStart, _this.options.scrollMargin, _this.options.getItemKey];
}, function(count, paddingStart, scrollMargin, getItemKey) {
_this.pendingMeasuredCacheIndexes = [];
return {
count,
paddingStart,
scrollMargin,
getItemKey
};
}, {
key: false
});
this.getFurthestMeasurement = function(measurements, index) {
var furthestMeasurementsFound = /* @__PURE__ */ new Map();
var furthestMeasurements = /* @__PURE__ */ new Map();
for (var m = index - 1; m >= 0; m--) {
var measurement = measurements[m];
if (furthestMeasurementsFound.has(measurement.lane)) {
continue;
}
var previousFurthestMeasurement = furthestMeasurements.get(measurement.lane);
if (previousFurthestMeasurement == null || measurement.end > previousFurthestMeasurement.end) {
furthestMeasurements.set(measurement.lane, measurement);
} else if (measurement.end < previousFurthestMeasurement.end) {
furthestMeasurementsFound.set(measurement.lane, true);
}
if (furthestMeasurementsFound.size === _this.options.lanes) {
break;
}
}
return furthestMeasurements.size === _this.options.lanes ? Array.from(furthestMeasurements.values()).sort(function(a, b) {
return a.end - b.end;
})[0] : void 0;
};
this.getMeasurements = memo(function() {
return [_this.memoOptions(), _this.itemSizeCache];
}, function(_ref4, itemSizeCache) {
var count = _ref4.count, paddingStart = _ref4.paddingStart, scrollMargin = _ref4.scrollMargin, getItemKey = _ref4.getItemKey;
var min = _this.pendingMeasuredCacheIndexes.length > 0 ? Math.min.apply(Math, _this.pendingMeasuredCacheIndexes) : 0;
_this.pendingMeasuredCacheIndexes = [];
var measurements = _this.measurementsCache.slice(0, min);
for (var _i2 = min; _i2 < count; _i2++) {
var key = getItemKey(_i2);
var furthestMeasurement = _this.options.lanes === 1 ? measurements[_i2 - 1] : _this.getFurthestMeasurement(measurements, _i2);
var start = furthestMeasurement ? furthestMeasurement.end : paddingStart + scrollMargin;
var measuredSize = itemSizeCache.get(key);
var size = typeof measuredSize === "number" ? measuredSize : _this.options.estimateSize(_i2);
var end = start + size;
var lane = furthestMeasurement ? furthestMeasurement.lane : _i2 % _this.options.lanes;
measurements[_i2] = {
index: _i2,
start,
size,
end,
key,
lane
};
}
_this.measurementsCache = measurements;
return measurements;
}, {
key: "getMeasurements",
debug: function debug() {
return _this.options.debug;
}
});
this.calculateRange = memo(function() {
return [_this.getMeasurements(), _this.getSize(), _this.scrollOffset];
}, function(measurements, outerSize, scrollOffset) {
return _this.range = calculateRange({
measurements,
outerSize,
scrollOffset
});
}, {
key: "calculateRange",
debug: function debug() {
return _this.options.debug;
}
});
this.maybeNotify = memo(function() {
var range = _this.calculateRange();
return [range.startIndex, range.endIndex, _this.isScrolling];
}, function() {
_this.notify();
}, {
key: "maybeNotify",
debug: function debug() {
return _this.options.debug;
},
initialDeps: [this.range.startIndex, this.range.endIndex, this.isScrolling]
});
this.getIndexes = memo(function() {
return [_this.options.rangeExtractor, _this.calculateRange(), _this.options.overscan, _this.options.count];
}, function(rangeExtractor, range, overscan, count) {
return rangeExtractor(_extends2({}, range, {
overscan,
count
}));
}, {
key: "getIndexes",
debug: function debug() {
return _this.options.debug;
}
});
this.indexFromElement = function(node) {
var attributeName = _this.options.indexAttribute;
var indexStr = node.getAttribute(attributeName);
if (!indexStr) {
console.warn("Missing attribute name '" + attributeName + "={index}' on measured element.");
return -1;
}
return parseInt(indexStr, 10);
};
this._measureElement = function(node, entry) {
var item = _this.measurementsCache[_this.indexFromElement(node)];
if (!item) {
_this.measureElementCache.forEach(function(cached, key) {
if (cached === node) {
_this.observer.unobserve(node);
_this.measureElementCache["delete"](key);
}
});
return;
}
var prevNode = _this.measureElementCache.get(item.key);
if (!node.isConnected) {
if (prevNode) {
_this.observer.unobserve(prevNode);
_this.measureElementCache["delete"](item.key);
}
return;
}
if (prevNode !== node) {
if (prevNode) {
_this.observer.unobserve(prevNode);
}
_this.observer.observe(node);
_this.measureElementCache.set(item.key, node);
}
var measuredItemSize = _this.options.measureElement(node, entry, _this);
_this.resizeItem(item, measuredItemSize);
};
this.resizeItem = function(item, size) {
var _this$itemSizeCache$g;
var itemSize = (_this$itemSizeCache$g = _this.itemSizeCache.get(item.key)) != null ? _this$itemSizeCache$g : item.size;
var delta = size - itemSize;
if (delta !== 0) {
if (item.start < _this.scrollOffset) {
if (_this.options.debug) {
console.info("correction", delta);
}
_this._scrollToOffset(_this.scrollOffset, {
adjustments: _this.scrollAdjustments += delta,
behavior: void 0
});
}
_this.pendingMeasuredCacheIndexes.push(item.index);
_this.itemSizeCache = new Map(_this.itemSizeCache.set(item.key, size));
_this.notify();
}
};
this.measureElement = function(node) {
if (!node) {
return;
}
_this._measureElement(node, void 0);
};
this.getVirtualItems = memo(function() {
return [_this.getIndexes(), _this.getMeasurements()];
}, function(indexes, measurements) {
var virtualItems = [];
for (var k = 0, len = indexes.length; k < len; k++) {
var _i3 = indexes[k];
var measurement = measurements[_i3];
virtualItems.push(measurement);
}
return virtualItems;
}, {
key: "getIndexes",
debug: function debug() {
return _this.options.debug;
}
});
this.getVirtualItemForOffset = function(offset) {
var measurements = _this.getMeasurements();
return notUndefined(measurements[findNearestBinarySearch(0, measurements.length - 1, function(index) {
return notUndefined(measurements[index]).start;
}, offset)]);
};
this.getOffsetForAlignment = function(toOffset, align) {
var size = _this.getSize();
if (align === "auto") {
if (toOffset <= _this.scrollOffset) {
align = "start";
} else if (toOffset >= _this.scrollOffset + size) {
align = "end";
} else {
align = "start";
}
}
if (align === "start") {
toOffset = toOffset;
} else if (align === "end") {
toOffset = toOffset - size;
} else if (align === "center") {
toOffset = toOffset - size / 2;
}
var scrollSizeProp = _this.options.horizontal ? "scrollWidth" : "scrollHeight";
var scrollSize = _this.scrollElement ? "document" in _this.scrollElement ? _this.scrollElement.document.documentElement[scrollSizeProp] : _this.scrollElement[scrollSizeProp] : 0;
var maxOffset = scrollSize - _this.getSize();
return Math.max(Math.min(maxOffset, toOffset), 0);
};
this.getOffsetForIndex = function(index, align) {
if (align === void 0) {
align = "auto";
}
index = Math.max(0, Math.min(index, _this.options.count - 1));
var measurement = notUndefined(_this.getMeasurements()[index]);
if (align === "auto") {
if (measurement.end >= _this.scrollOffset + _this.getSize() - _this.options.scrollPaddingEnd) {
align = "end";
} else if (measurement.start <= _this.scrollOffset + _this.options.scrollPaddingStart) {
align = "start";
} else {
return [_this.scrollOffset, align];
}
}
var toOffset = align === "end" ? measurement.end + _this.options.scrollPaddingEnd : measurement.start - _this.options.scrollPaddingStart;
return [_this.getOffsetForAlignment(toOffset, align), align];
};
this.isDynamicMode = function() {
return _this.measureElementCache.size > 0;
};
this.cancelScrollToIndex = function() {
if (_this.scrollToIndexTimeoutId !== null) {
clearTimeout(_this.scrollToIndexTimeoutId);
_this.scrollToIndexTimeoutId = null;
}
};
this.scrollToOffset = function(toOffset, _temp) {
var _ref5 = _temp === void 0 ? {} : _temp, _ref5$align = _ref5.align, align = _ref5$align === void 0 ? "start" : _ref5$align, behavior = _ref5.behavior;
_this.cancelScrollToIndex();
if (behavior === "smooth" && _this.isDynamicMode()) {
console.warn("The `smooth` scroll behavior is not fully supported with dynamic size.");
}
_this._scrollToOffset(_this.getOffsetForAlignment(toOffset, align), {
adjustments: void 0,
behavior
});
};
this.scrollToIndex = function(index, _temp2) {
var _ref6 = _temp2 === void 0 ? {} : _temp2, _ref6$align = _ref6.align, initialAlign = _ref6$align === void 0 ? "auto" : _ref6$align, behavior = _ref6.behavior;
index = Math.max(0, Math.min(index, _this.options.count - 1));
_this.cancelScrollToIndex();
if (behavior === "smooth" && _this.isDynamicMode()) {
console.warn("The `smooth` scroll behavior is not fully supported with dynamic size.");
}
var _this$getOffsetForInd = _this.getOffsetForIndex(index, initialAlign), toOffset = _this$getOffsetForInd[0], align = _this$getOffsetForInd[1];
_this._scrollToOffset(toOffset, {
adjustments: void 0,
behavior
});
if (behavior !== "smooth" && _this.isDynamicMode()) {
_this.scrollToIndexTimeoutId = setTimeout(function() {
_this.scrollToIndexTimeoutId = null;
var elementInDOM = _this.measureElementCache.has(_this.options.getItemKey(index));
if (elementInDOM) {
var _this$getOffsetForInd2 = _this.getOffsetForIndex(index, align), _toOffset = _this$getOffsetForInd2[0];
if (!approxEqual(_toOffset, _this.scrollOffset)) {
_this.scrollToIndex(index, {
align,
behavior
});
}
} else {
_this.scrollToIndex(index, {
align,
behavior
});
}
});
}
};
this.scrollBy = function(delta, _temp3) {
var _ref7 = _temp3 === void 0 ? {} : _temp3, behavior = _ref7.behavior;
_this.cancelScrollToIndex();
if (behavior === "smooth" && _this.isDynamicMode()) {
console.warn("The `smooth` scroll behavior is not fully supported with dynamic size.");
}
_this._scrollToOffset(_this.scrollOffset + delta, {
adjustments: void 0,
behavior
});
};
this.getTotalSize = function() {
var _this$getMeasurements;
return (((_this$getMeasurements = _this.getMeasurements()[_this.options.count - 1]) == null ? void 0 : _this$getMeasurements.end) || _this.options.paddingStart) - _this.options.scrollMargin + _this.options.paddingEnd;
};
this._scrollToOffset = function(offset, _ref8) {
var adjustments = _ref8.adjustments, behavior = _ref8.behavior;
_this.options.scrollToFn(offset, {
behavior,
adjustments
}, _this);
};
this.measure = function() {
_this.itemSizeCache = /* @__PURE__ */ new Map();
_this.notify();
};
this.setOptions(_opts);
this.scrollRect = this.options.initialRect;
this.scrollOffset = this.options.initialOffset;
this.measurementsCache = this.options.initialMeasurementsCache;
this.measurementsCache.forEach(function(item) {
_this.itemSizeCache.set(item.key, item.size);
});
this.maybeNotify();
};
var findNearestBinarySearch = function findNearestBinarySearch2(low, high, getCurrentValue, value) {
while (low <= high) {
var middle = (low + high) / 2 | 0;
var currentValue = getCurrentValue(middle);
if (currentValue < value) {
low = middle + 1;
} else if (currentValue > value) {
high = middle - 1;
} else {
return middle;
}
}
if (low > 0) {
return low - 1;
} else {
return 0;
}
};
function calculateRange(_ref9) {
var measurements = _ref9.measurements, outerSize = _ref9.outerSize, scrollOffset = _ref9.scrollOffset;
var count = measurements.length - 1;
var getOffset = function getOffset2(index) {
return measurements[index].start;
};
var startIndex = findNearestBinarySearch(0, count, getOffset, scrollOffset);
var endIndex = startIndex;
while (endIndex < count && measurements[endIndex].end < scrollOffset + outerSize) {
endIndex++;
}
return {
startIndex,
endIndex
};
}
// ../../node_modules/@tanstack/react-virtual/build/lib/index.mjs
var useIsomorphicLayoutEffect = typeof document !== "undefined" ? React.useLayoutEffect : React.useEffect;
function useVirtualizerBase(options) {
var rerender = React.useReducer(function() {
return {};
}, {})[1];
var resolvedOptions = _extends({}, options, {
onChange: function onChange(instance2) {
rerender();
options.onChange == null ? void 0 : options.onChange(instance2);
}
});
var _React$useState = React.useState(function() {
return new Virtualizer(resolvedOptions);
}), instance = _React$useState[0];
instance.setOptions(resolvedOptions);
React.useEffect(function() {
return instance._didMount();
}, []);
useIsomorphicLayoutEffect(function() {
return instance._willUpdate();
});
return instance;
}
function useVirtualizer(options) {
return useVirtualizerBase(_extends({
observeElementRect,
observeElementOffset,
scrollToFn: elementScroll
}, options));
}
// src/components/combobox/combobox.tsx
var import_react19 = __toESM(require("react"), 1);
// src/hooks/use-computed.ts
var import_react3 = require("react");
// src/hooks/use-iso-morphic-effect.ts
var import_react = require("react");
// src/utils/env.ts
var Env = class {
constructor() {
__publicField(this, "current", this.detect());
__publicField(this, "handoffState", "pending");
__publicField(this, "currentId", 0);
}
set(env2) {
if (this.current === env2)
return;
this.handoffState = "pending";
this.currentId = 0;
this.current = env2;
}
reset() {
this.set(this.detect());
}
nextId() {
return ++this.currentId;
}
get isServer() {
return this.current === "server";
}
get isClient() {
return this.current === "client";
}
detect() {
if (typeof window === "undefined" || typeof document === "undefined") {
return "server";
}
return "client";
}
handoff() {
if (this.handoffState === "pending") {
this.handoffState = "complete";
}
}
get isHandoffComplete() {
return this.handoffState === "complete";
}
};
var env = new Env();
// src/hooks/use-iso-morphic-effect.ts
var useIsoMorphicEffect = (effect, deps) => {
if (env.isServer) {
(0, import_react.useEffect)(effect, deps);
} else {
(0, import_react.useLayoutEffect)(effect, deps);
}
};
// src/hooks/use-latest-value.ts
var import_react2 = require("react");
function useLatestValue(value) {
let cache = (0, import_react2.useRef)(value);
useIsoMorphicEffect(() => {
cache.current = value;
}, [value]);
return cache;
}
// src/hooks/use-computed.ts
function useComputed(cb, dependencies) {
let [value, setValue] = (0, import_react3.useState)(cb);
let cbRef = useLatestValue(cb);
useIsoMorphicEffect(() => setValue(cbRef.current), [cbRef, setValue, ...dependencies]);
return value;
}
// src/hooks/use-controllable.ts
var import_react5 = require("react");
// src/hooks/use-event.ts
var import_react4 = __toESM(require("react"), 1);
var useEvent = (
// TODO: Add React.useEvent ?? once the useEvent hook is available
function useEvent2(cb) {
let cache = useLatestValue(cb);
return import_react4.default.useCallback((...args) => cache.current(...args), [cache]);
}
);
// src/hooks/use-controllable.ts
function useControllable(controlledValue, onChange, defaultValue) {
let [internalValue, setInternalValue] = (0, import_react5.useState)(defaultValue);
let isControlled = controlledValue !== void 0;
let wasControlled = (0, import_react5.useRef)(isControlled);
let didWarnOnUncontrolledToControlled = (0, import_react5.useRef)(false);
let didWarnOnControlledToUncontrolled = (0, import_react5.useRef)(false);
if (isControlled && !wasControlled.current && !didWarnOnUncontrolledToControlled.current) {
didWarnOnUncontrolledToControlled.current = true;
wasControlled.current = isControlled;
console.error(
"A component is changing from uncontrolled to controlled. This may be caused by the value changing from undefined to a defined value, which should not happen."
);
} else if (!isControlled && wasControlled.current && !didWarnOnControlledToUncontrolled.current) {
didWarnOnControlledToUncontrolled.current = true;
wasControlled.current = isControlled;
console.error(
"A component is changing from controlled to uncontrolled. This may be caused by the value changing from a defined value to undefined, which should not happen."
);
}
return [
isControlled ? controlledValue : internalValue,
useEvent((value) => {
if (isControlled) {
return onChange == null ? void 0 : onChange(value);
} else {
setInternalValue(value);
return onChange == null ? void 0 : onChange(value);
}
})
];
}
// src/hooks/use-disposables.ts
var import_react6 = require("react");
// src/utils/micro-task.ts
function microTask(cb) {
if (typeof queueMicrotask === "function") {
queueMicrotask(cb);
} else {
Promise.resolve().then(cb).catch(
(e) => setTimeout(() => {
throw e;
})
);
}
}
// src/utils/disposables.ts
function disposables() {
let _disposables = [];
let api = {
addEventListener(element, name, listener, options) {
element.addEventListener(name, listener, options);
return api.add(() => element.removeEventListener(name, listener, options));
},
requestAnimationFrame(...args) {
let raf = requestAnimationFrame(...args);
return api.add(() => cancelAnimationFrame(raf));
},
nextFrame(...args) {
return api.requestAnimationFrame(() => {
return api.requestAnimationFrame(...args);
});
},
setTimeout(...args) {
let timer = setTimeout(...args);
return api.add(() => clearTimeout(timer));
},
microTask(...args) {
let task = { current: true };
microTask(() => {
if (task.current) {
args[0]();
}
});
return api.add(() => {
task.current = false;
});
},
style(node, property, value) {
let previous = node.style.getPropertyValue(property);
Object.assign(node.style, { [property]: value });
return this.add(() => {
Object.assign(node.style, { [property]: previous });
});
},
group(cb) {
let d = disposables();
cb(d);
return this.add(() => d.dispose());
},
add(cb) {
_disposables.push(cb);
return () => {
let idx = _disposables.indexOf(cb);
if (idx >= 0) {
for (let dispose of _disposables.splice(idx, 1)) {
dispose();
}
}
};
},
dispose() {
for (let dispose of _disposables.splice(0)) {
dispose();
}
}
};
return api;
}
// src/hooks/use-disposables.ts
function useDisposables() {
let [d] = (0, import_react6.useState)(disposables);
(0, import_react6.useEffect)(() => () => d.dispose(), [d]);
return d;
}
// src/hooks/use-id.ts
var import_react7 = __toESM(require("react"), 1);
// src/hooks/use-server-handoff-complete.ts
var React3 = __toESM(require("react"), 1);
function useIsHydratingInReact18() {
let isServer = typeof document === "undefined";
if (!("useSyncExternalStore" in React3)) {
return false;
}
const useSyncExternalStore4 = ((r) => r.useSyncExternalStore)(React3);
let result = useSyncExternalStore4(
() => () => {
},
() => false,
() => isServer ? false : true
);
return result;
}
function useServerHandoffComplete() {
let isHydrating = useIsHydratingInReact18();
let [complete, setComplete] = React3.useState(env.isHandoffComplete);
if (complete && env.isHandoffComplete === false) {
setComplete(false);
}
React3.useEffect(() => {
if (complete === true)
return;
setComplete(true);
}, [complete]);
React3.useEffect(() => env.handoff(), []);
if (isHydrating) {
return false;
}
return complete;
}
// src/hooks/use-id.ts
var _a;
var useId = (
// Prefer React's `useId` if it's available.
// @ts-expect-error - `useId` doesn't exist in React < 18.
(_a = import_react7.default.useId) != null ? _a : function useId2() {
let ready = useServerHandoffComplete();
let [id, setId] = import_react7.default.useState(ready ? () => env.nextId() : null);
useIsoMorphicEffect(() => {
if (id === null)
setId(env.nextId());
}, [id]);
return id != null ? "" + id : void 0;
}
);
// src/hooks/use-outside-click.ts
var import_react10 = require("react");
// src/utils/match.ts
function match(value, lookup, ...args) {
if (value in lookup) {
let returnValue = lookup[value];
return typeof returnValue === "function" ? returnValue(...args) : returnValue;
}
let error = new Error(
`Tried to handle "${value}" but there is no handler defined. Only defined handlers are: ${Object.keys(
lookup
).map((key) => `"${key}"`).join(", ")}.`
);
if (Error.captureStackTrace)
Error.captureStackTrace(error, match);
throw error;
}
// src/utils/owner.ts
function getOwnerDocument(element) {
if (env.isServer)
return null;
if (element instanceof Node)
return element.ownerDocument;
if (element == null ? void 0 : element.hasOwnProperty("current")) {
if (element.current instanceof Node)
return element.current.ownerDocument;
}
return document;
}
// src/utils/focus-management.ts
var focusableSelector = [
"[contentEditable=true]",
"[tabindex]",
"a[href]",
"area[href]",
"button:not([disabled])",
"iframe",
"input:not([disabled])",
"select:not([disabled])",
"textarea:not([disabled])"
].map(
false ? (
// TODO: Remove this once JSDOM fixes the issue where an element that is
// "hidden" can be the document.activeElement, because this is not possible
// in real browsers.
(selector) => `${selector}:not([tabindex='-1']):not([style*='display: none'])`
) : (selector) => `${selector}:not([tabindex='-1'])`
).join(",");
function getFocusableElements(container = document.body) {
if (container == null)
return [];
return Array.from(container.querySelectorAll(focusableSelector)).sort(
// We want to move `tabIndex={0}` to the end of the list, this is what the browser does as well.
(a, z) => Math.sign((a.tabIndex || Number.MAX_SAFE_INTEGER) - (z.tabIndex || Number.MAX_SAFE_INTEGER))
);
}
function isFocusableElement(element, mode = 0 /* Strict */) {
var _a3;
if (element === ((_a3 = getOwnerDocument(element)) == null ? void 0 : _a3.body))
return false;
return match(mode, {
[0 /* Strict */]() {
return element.matches(focusableSelector);
},
[1 /* Loose */]() {
let next = element;
while (next !== null) {
if (next.matches(focusableSelector))
return true;
next = next.parentElement;
}
return false;
}
});
}
function restoreFocusIfNecessary(element) {
let ownerDocument = getOwnerDocument(element);
disposables().nextFrame(() => {
if (ownerDocument && !isFocusableElement(ownerDocument.activeElement, 0 /* Strict */)) {
focusElement(element);
}
});
}
if (typeof window !== "undefined" && typeof document !== "undefined") {
document.addEventListener(
"keydown",
(event) => {
if (event.metaKey || event.altKey || event.ctrlKey) {
return;
}
document.documentElement.dataset.headlessuiFocusVisible = "";
},
true
);
document.addEventListener(
"click",
(event) => {
if (event.detail === 1 /* Mouse */) {
delete document.documentElement.dataset.headlessuiFocusVisible;
} else if (event.detail === 0 /* Keyboard */) {
document.documentElement.dataset.headlessuiFocusVisible = "";
}
},
true
);
}
function focusElement(element) {
element == null ? void 0 : element.focus({ preventScroll: true });
}
var selectableSelector = ["textarea", "input"].join(",");
function isSelectableElement(element) {
var _a3, _b;
return (_b = (_a3 = element == null ? void 0 : element.matches) == null ? void 0 : _a3.call(element, selectableSelector)) != null ? _b : false;
}
function sortByDomNode(nodes, resolveKey = (i) => i) {
return nodes.slice().sort((aItem, zItem) => {
let a = resolveKey(aItem);
let z = resolveKey(zItem);
if (a === null || z === null)
return 0;
let position = a.compareDocumentPosition(z);
if (position & Node.DOCUMENT_POSITION_FOLLOWING)
return -1;
if (position & Node.DOCUMENT_POSITION_PRECEDING)
return 1;
return 0;
});
}
function focusFrom(current, focus) {
return focusIn(getFocusableElements(), focus, { relativeTo: current });
}
function focusIn(container, focus, {
sorted = true,
relativeTo = null,
skipElements = []
} = {}) {
let ownerDocument = Array.isArray(container) ? container.length > 0 ? container[0].ownerDocument : document : container.ownerDocument;
let elements = Array.isArray(container) ? sorted ? sortByDomNode(container) : container : getFocusableElements(container);
if (skipElements.length > 0 && elements.length > 1) {
elements = elements.filter((x) => !skipElements.includes(x));
}
relativeTo = relativeTo != null ? relativeTo : ownerDocument.activeElement;
let direction = (() => {
if (focus & (1 /* First */ | 4 /* Next */))
return 1 /* Next */;
if (focus & (2 /* Previous */ | 8 /* Last */))
return -1 /* Previous */;
throw new Error("Missing Focus.First, Focus.Previous, Focus.Next or Focus.Last");
})();
let startIndex = (() => {
if (focus & 1 /* First */)
return 0;
if (focus & 2 /* Previous */)
return Math.max(0, elements.indexOf(relativeTo)) - 1;
if (focus & 4 /* Next */)
return Math.max(0, elements.indexOf(relativeTo)) + 1;
if (focus & 8 /* Last */)
return elements.length - 1;
throw new Error("Missing Focus.First, Focus.Previous, Focus.Next or Focus.Last");
})();
let focusOptions = focus & 32 /* NoScroll */ ? { preventScroll: true } : {};
let offset = 0;
let total = elements.length;
let next = void 0;
do {
if (offset >= total || offset + total <= 0)
return 0 /* Error */;
let nextIdx = startIndex + offset;
if (focus & 16 /* WrapAround */) {
nextIdx = (nextIdx + total) % total;
} else {
if (nextIdx < 0)
return 3 /* Underflow */;
if (nextIdx >= total)
return 1 /* Overflow */;
}
next = elements[nextIdx];
next == null ? void 0 : next.focus(focusOptions);
offset += direction;
} while (next !== ownerDocument.activeElement);
if (focus & (4 /* Next */ | 2 /* Previous */) && isSelectableElement(next)) {
next.select();
}
return 2 /* Success */;
}
// src/utils/platform.ts
function isIOS() {
return (
// Check if it is an iPhone
/iPhone/gi.test(window.navigator.platform) || // Check if it is an iPad. iPad reports itself as "MacIntel", but we can check if it is a touch
// screen. Let's hope that Apple doesn't release a touch screen Mac (or maybe this would then
// work as expected 🤔).
/Mac/gi.test(window.navigator.platform) && window.navigator.maxTouchPoints > 0
);
}
function isAndroid() {
return /Android/gi.test(window.navigator.userAgent);
}
function isMobile() {
return isIOS() || isAndroid();
}
// src/hooks/use-document-event.ts
var import_react8 = require("react");
function useDocumentEvent(type, listener, options) {
let listenerRef = useLatestValue(listener);
(0, import_react8.useEffect)(() => {
function handler(event) {
listenerRef.current(event);
}
document.addEventListener(type, handler, options);
return () => document.removeEventListener(type, handler, options);
}, [type, options]);
}
// src/hooks/use-window-event.ts
var import_react9 = require("react");
function useWindowEvent(type, listener, options) {
let listenerRef = useLatestValue(listener);
(0, import_react9.useEffect)(() => {
function handler(event) {
listenerRef.current(event);
}
window.addEventListener(type, handler, options);
return () => window.removeEventListener(type, handler, options);
}, [type, options]);
}
// src/hooks/use-outside-click.ts
function useOutsideClick(containers, cb, enabled = true) {
let enabledRef = (0, import_react10.useRef)(false);
(0, import_react10.useEffect)(
false ? () => {
enabledRef.current = enabled;
} : () => {
requestAnimationFrame(() => {
enabledRef.current = enabled;
});
},
[enabled]
);
function handleOutsideClick(event, resolveTarget) {
if (!enabledRef.current)
return;
if (event.defaultPrevented)
return;
let target = resolveTarget(event);
if (target === null) {
return;
}
if (!target.getRootNode().contains(target))
return;
if (!target.isConnected)
return;
let _containers = function resolve(containers2) {
if (typeof containers2 === "function") {
return resolve(containers2());
}
if (Array.isArray(containers2)) {
return containers2;
}
if (containers2 instanceof Set) {
return containers2;
}
return [containers2];
}(containers);
for (let container of _containers) {
if (container === null)
continue;
let domNode = container instanceof HTMLElement ? container : container.current;
if (domNode == null ? void 0 : domNode.contains(target)) {
return;
}
if (event.composed && event.composedPath().includes(domNode)) {
return;
}
}
if (
// This check alllows us to know whether or not we clicked on a "focusable" element like a
// button or an input. This is a backwards compatibility check so that you can open a <Menu
// /> and click on another <Menu /> which should close Menu A and open Menu B. We might
// revisit that so that you will require 2 clicks instead.
!isFocusableElement(target, 1 /* Loose */) && // This could be improved, but the `Combobox.Button` adds tabIndex={-1} to make it
// unfocusable via the keyboard so that tabbing to the next item from the input doesn't
// first go to the button.
target.tabIndex !== -1
) {
event.preventDefault();
}
return cb(event, target);
}
let initialClickTarget = (0, import_react10.useRef)(null);
useDocumentEvent(
"pointerdown",
(event) => {
var _a3, _b;
if (enabledRef.current) {
initialClickTarget.current = ((_b = (_a3 = event.composedPath) == null ? void 0 : _a3.call(event)) == null ? void 0 : _b[0]) || event.target;
}
},
true
);
useDocumentEvent(
"mousedown",
(event) => {
var _a3, _b;
if (enabledRef.current) {
initialClickTarget.current = ((_b = (_a3 = event.composedPath) == null ? void 0 : _a3.call(event)) == null ? void 0 : _b[0]) || event.target;
}
},
true
);
useDocumentEvent(
"click",
(event) => {
if (isMobile()) {
return;
}
if (!initialClickTarget.current) {
return;
}
handleOutsideClick(event, () => {
return initialClickTarget.current;
});
initialClickTarget.current = null;
},
// We will use the `capture` phase so that layers in between with `event.stopPropagation()`
// don't "cancel" this outside click check. E.g.: A `Menu` inside a `DialogPanel` if the `Menu`
// is open, and you click outside of it in the `DialogPanel` the `Menu` should close. However,
// the `DialogPanel` has a `onClick(e) { e.stopPropagation() }` which would cancel this.
true
);
useDocumentEvent(
"touchend",
(event) => {
return handleOutsideClick(event, () => {
if (event.target instanceof HTMLElement) {
return event.target;
}
return null;
});
},
// We will use the `capture` phase so that layers in between with `event.stopPropagation()`
// don't "cancel" this outside click check. E.g.: A `Menu` inside a `DialogPanel` if the `Menu`
// is open, and you click outside of it in the `DialogPanel` the `Menu` should close. However,
// the `DialogPanel` has a `onClick(e) { e.stopPropagation() }` which would cancel this.
true
);
useWindowEvent(
"blur",
(event) => {
return handleOutsideClick(event, () => {
return window.document.activeElement instanceof HTMLIFrameElement ? window.document.activeElement : null;
});
},
true
);
}
// src/hooks/use-owner.ts
var import_react11 = require("react");
function useOwnerDocument(...args) {
return (0, import_react11.useMemo)(() => getOwnerDocument(...args), [...args]);
}
// src/hooks/use-resolve-button-type.ts
var import_react12 = require("react");
function resolveType(props) {
var _a3;
if (props.type)
return props.type;
let tag = (_a3 = props.as) != null ? _a3 : "button";
if (typeof tag === "string" && tag.toLowerCase() === "button")
return "button";
return void 0;
}
function useResolveButtonType(props, ref) {
let [type, setType] = (0, import_react12.useState)(() => resolveType(props));
useIsoMorphicEffect(() => {
setType(resolveType(props));
}, [props.type, props.as]);
useIsoMorphicEffect(() => {
if (type)
return;
if (!ref.current)
return;
if (ref.current instanceof HTMLButtonElement && !ref.current.hasAttribute("type")) {
setType("button");
}
}, [type, ref]);
return type;
}
// src/hooks/use-sync-refs.ts
var import_react13 = require("react");
var Optional = Symbol();
function optionalRef(cb, isOptional = true) {
return Object.assign(cb, { [Optional]: isOptional });
}
function useSyncRefs(...refs) {
let cache = (0, import_react13.useRef)(refs);
(0, import_react13.useEffect)(() => {
cache.current = refs;
}, [refs]);
let syncRefs = useEvent((value) => {
for (let ref of cache.current) {
if (ref == null)
continue;
if (typeof ref === "function")
ref(value);
else
ref.current = value;
}
});
return refs.every(
(ref) => ref == null || // @ts-expect-error
(ref == null ? void 0 : ref[Optional])
) ? void 0 : syncRefs;
}
// src/hooks/use-tracked-pointer.ts
var import_react14 = require("react");
function eventToPosition(evt) {
return [evt.screenX, evt.screenY];
}
function useTrackedPointer() {
let lastPos = (0, import_react14.useRef)([-1, -1]);
return {
wasMoved(evt) {
if (false) {
return true;
}
let newPos = eventToPosition(evt);
if (lastPos.current[0] === newPos[0] && lastPos.current[1] === newPos[1]) {
return false;
}
lastPos.current = newPos;
return true;
},
update(evt) {
lastPos.current = eventToPosition(evt);
}
};
}
// src/hooks/use-tree-walker.ts
var import_react15 = require("react");
function useTreeWalker({
container,
accept,
walk,
enabled = true
}) {
let acceptRef = (0, import_react15.useRef)(accept);
let walkRef = (0, import_react15.useRef)(walk);
(0, import_react15.useEffect)(() => {
acceptRef.current = accept;
walkRef.current = walk;
}, [accept, walk]);
useIsoMorphicEffect(() => {
if (!container)
return;
if (!enabled)
return;
let ownerDocument = getOwnerDocument(container);
if (!ownerDocument)
return;
let accept2 = acceptRef.current;
let walk2 = walkRef.current;
let acceptNode = Object.assign((node) => accept2(node), { acceptNode: accept2 });
let walker = ownerDocument.createTreeWalker(
container,
NodeFilter.SHOW_ELEMENT,
acceptNode,
// @ts-expect-error This `false` is a simple small fix for older browsers
false
);
while (walker.nextNode())
walk2(walker.currentNode);
}, [container, enabled, acceptRef, walkRef]);
}
// src/hooks/use-watch.ts
var import_react16 = require("react");
function useWatch(cb, dependencies) {
let track = (0, import_react16.useRef)([]);
let action = useEvent(cb);
(0, import_react16.useEffect)(() => {
let oldValues = [...track.current];
for (let [idx, value] of dependencies.entries()) {
if (track.current[idx] !== value) {
let returnValue = action(dependencies, oldValues);
track.current = dependencies;
return returnValue;
}
}
}, [action, ...dependencies]);
}
// src/utils/render.ts
var import_react17 = require("react");
// src/utils/class-names.ts
function classNames(...classes) {
return Array.from(
new Set(
classes.flatMap((value) => {
if (typeof value === "string") {
return value.split(" ");
}
return [];
})
)
).filter(Boolean).join(" ");
}
// src/utils/render.ts
function render({
ourProps,
theirProps,
slot,
defaultTag,
features,
visible = true,
name,
mergeRefs
}) {
mergeRefs = mergeRefs != null ? mergeRefs : defaultMergeRefs;
let props = mergeProps(theirProps, ourProps);
if (visible)
return _render(props, slot, defaultTag, name, mergeRefs);
let featureFlags = features != null ? features : 0 /* None */;
if (featureFlags & 2 /* Static */) {
let { static: isStatic = false, ...rest } = props;
if (isStatic)
return _render(rest, slot, defaultTag, name, mergeRefs);
}
if (featureFlags & 1 /* RenderStrategy */) {
let { unmount = true, ...rest } = props;
let strategy = unmount ? 0 /* Unmount */ : 1 /* Hidden */;
return match(strategy, {
[0 /* Unmount */]() {
return null;
},
[1 /* Hidden */]() {
return _render(
{ ...rest, ...{ hidden: true, style: { display: "none" } } },
slot,
defaultTag,
name,
mergeRefs
);
}
});
}
return _render(props, slot, defaultTag, name, mergeRefs);
}
function _render(props, slot = {}, tag, name, mergeRefs) {
let {
as: Component = tag,
children,
refName = "ref",
...rest
} = omit(props, ["unmount", "static"]);
let refRelatedProps = props.ref !== void 0 ? { [refName]: props.ref } : {};
let resolvedChildren = typeof children === "function" ? children(slot) : children;
if ("className" in rest && rest.className && typeof rest.className === "function") {
rest.className = rest.className(slot);
}
let dataAttributes = {};
if (slot) {
let exposeState = false;
let states = [];
for (let [k, v] of Object.entries(slot)) {
if (typeof v === "boolean") {
exposeState = true;
}
if (v === true) {
states.push(k);
}
}
if (exposeState)
dataAttributes[`data-headlessui-state`] = states.join(" ");
}
if (Component === import_react17.Fragment) {
if (Object.keys(compact(rest)).length > 0) {
if (!(0, import_react17.isValidElement)(resolvedChildren) || Array.isArray(resolvedChildren) && resolvedChildren.length > 1) {
throw new Error(
[
'Passing props on "Fragment"!',
"",
`The current component <${name} /> is rendering a "Fragment".`,
`However we need to passthrough the following props:`,
Object.keys(rest).map((line) => ` - ${line}`).join("\n"),
"",
"You can apply a few solutions:",
[
'Add an `as="..."` prop, to ensure that we render an actual element instead of a "Fragment".',
"Render a single element as the child so that we can forward the props onto that element."
].map((line) => ` - ${line}`).join("\n")
].join("\n")
);
}
let childProps = resolvedChildren.props;
let newClassName = typeof (childProps == null ? void 0 : childProps.className) === "function" ? (...args) => classNames(childProps == null ? void 0 : childProps.className(...args), rest.className) : classNames(childProps == null ? void 0 : childProps.className, rest.className);
let classNameProps = newClassName ? { className: newClassName } : {};
return (0, import_react17.cloneElement)(
resolvedChildren,
Object.assign(
{},
// Filter out undefined values so that they don't override the existing values
mergeProps(resolvedChildren.props, compact(omit(rest, ["ref"]))),
dataAttributes,
refRelatedProps,
{ ref: mergeRefs(resolvedChildren.ref, refRelatedProps.ref) },
classNameProps
)
);
}
}
return (0, import_react17.createElement)(
Component,
Object.assign(
{},
omit(rest, ["ref"]),
Component !== import_react17.Fragment && refRelatedProps,
Component !== import_react17.Fragment && dataAttributes
),
resolvedChildren
);
}
function useMergeRefsFn() {
let currentRefs = (0, import_react17.useRef)([]);
let mergedRef = (0, import_react17.useCallback)((value) => {
for (let ref of currentRefs.current) {
if (ref == null)
continue;
if (typeof ref === "function")
ref(value);
else
ref.current = value;
}
}, []);
return (...refs) => {
if (refs.every((ref) => ref == null)) {
return void 0;
}
currentRefs.current = refs;
return mergedRef;
};
}
function defaultMergeRefs(...refs) {
return refs.every((ref) => ref == null) ? void 0 : (value) => {
for (let ref of refs) {
if (ref == null)
continue;
if (typeof ref === "function")
ref(value);
else
ref.current = value;
}
};
}
function mergeProps(...listOfProps) {
var _a3;
if (listOfProps.length === 0)
return {};
if (listOfProps.length === 1)
return listOfProps[0];
let target = {};
let eventHandlers = {};
for (let props of listOfProps) {
for (let prop in props) {
if (prop.startsWith("on") && typeof props[prop] === "function") {
(_a3 = eventHandlers[prop]) != null ? _a3 : eventHandlers[prop] = [];
eventHandlers[prop].push(props[prop]);
} else {
target[prop] = props[prop];
}
}
}
if (target.disabled || target["aria-disabled"]) {
return Object.assign(
target,
// Set all event listeners that we collected to `undefined`. This is
// important because of the `cloneElement` from above, which merges the
// existing and new props, they don't just override therefore we have to
// explicitly nullify them.
Object.fromEntries(Object.keys(eventHandlers).map((eventName) => [eventName, void 0]))
);
}
for (let eventName in eventHandlers) {
Object.assign(target, {
[eventName](event, ...args) {
let handlers = eventHandlers[eventName];
for (let handler of handlers) {
if ((event instanceof Event || (event == null ? void 0 : event.nativeEvent) instanceof Event) && event.defaultPrevented) {
return;
}
handler(event, ...args);
}
}
});
}
return target;
}
function forwardRefWithAs(component) {
var _a3;
return Object.assign((0, import_react17.forwardRef)(component), {
displayName: (_a3 = component.displayName) != null ? _a3 : component.name
});
}
function compact(object) {
let clone = Object.assign({}, object);
for (let key in clone) {
if (clone[key] === void 0)
delete clone[key];
}
return clone;
}
function omit(object, keysToOmit = []) {
let clone = Object.assign({}, object);
for (let key of keysToOmit) {
if (key in clone)
delete clone[key];
}
return clone;
}
// src/internal/hidden.tsx
var DEFAULT_VISUALLY_HIDDEN_TAG = "div";
function VisuallyHidden(props, ref) {
var _a3;
let { features = 1 /* None */, ...theirProps } = props;
let ourProps = {
ref,
"aria-hidden": (features & 2 /* Focusable */) === 2 /* Focusable */ ? true : (_a3 = theirProps["aria-hidden"]) != null ? _a3 : void 0,
hidden: (features & 4 /* Hidden */) === 4 /* Hidden */ ? true : void 0,
style: {
position: "fixed",
top: 1,
left: 1,
width: 1,
height: 0,
padding: 0,
margin: -1,
overflow: "hidden",
clip: "rect(0, 0, 0, 0)",
whiteSpace: "nowrap",
borderWidth: "0",
...(features & 4 /* Hidden */) === 4 /* Hidden */ && !((features & 2 /* Focusable */) === 2 /* Focusable */) && { display: "none" }
}
};
return render({
ourProps,
theirProps,
slot: {},
defaultTag: DEFAULT_VISUALLY_HIDDEN_TAG,
name: "Hidden"
});
}
var Hidden = forwardRefWithAs(VisuallyHidden);
// src/internal/open-closed.tsx
var import_react18 = __toESM(require("react"), 1);
var Context = (0, import_react18.createContext)(null);
Context.displayName = "OpenClosedContext";
function useOpenClosed() {
return (0, import_react18.useContext)(Context);
}
function OpenClosedProvider({ value, children }) {
return /* @__PURE__ */ import_react18.default.createElement(Context.Provider, { value }, children);
}
// src/utils/document-ready.ts
function onDocumentReady(cb) {
function check() {
if (document.readyState === "loading")
return;
cb();
document.removeEventListener("DOMContentLoaded", check);
}
if (typeof window !== "undefined" && typeof document !== "undefined") {
document.addEventListener("DOMContentLoaded", check);
check();
}
}
// src/utils/active-element-history.ts
var history = [];
onDocumentReady(() => {
function handle(e) {
if (!(e.target instanceof HTMLElement))
return;
if (e.target === document.body)
return;
if (history[0] === e.target)
return;
history.unshift(e.target);
history = history.filter((x) => x != null && x.isConnected);
history.splice(10);
}
window.addEventListener("click", handle, { capture: true });
window.addEventListener("mousedown", handle, { capture: true });
window.addEventListener("focus", handle, { capture: true });
document.body.addEventListener("click", handle, { capture: true });
document.body.addEventListener("mousedown", handle, { capture: true });
document.body.addEventListener("focus", handle, { capture: true });
});
// src/utils/bugs.ts
function isDisabledReactIssue7711(element) {
let parent = element.parentElement;
let legend = null;
while (parent && !(parent instanceof HTMLFieldSetElement)) {
if (parent instanceof HTMLLegendElement)
legend = parent;
parent = parent.parentElement;
}
let isParentDisabled = (parent == null ? void 0 : parent.getAttribute("disabled")) === "";
if (isParentDisabled && isFirstLegend(legend))
return false;
return isParentDisabled;
}
function isFirstLegend(element) {
if (!element)
return false;
let previous = element.previousElementSibling;
while (previous !== null) {
if (previous instanceof HTMLLegendElement)
return false;
previous = previous.previousElementSibling;
}
return true;
}
// src/utils/calculate-active-index.ts
function assertNever(x) {
throw new Error("Unexpected object: " + x);
}
function calculateActiveIndex(action, resolvers) {
let items = resolvers.resolveItems();
if (items.length <= 0)
return null;
let currentActiveIndex = resolvers.resolveActiveIndex();
let activeIndex = currentActiveIndex != null ? currentActiveIndex : -1;
switch (action.focus) {
case 0 /* First */: {
for (let i = 0; i < items.length; ++i) {
if (!resolvers.resolveDisabled(items[i], i, items)) {
return i;
}
}
return currentActiveIndex;
}
case 1 /* Previous */: {
for (let i = activeIndex - 1; i >= 0; --i) {
if (!resolvers.resolveDisabled(items[i], i, items)) {
return i;
}
}
return currentActiveIndex;
}
case 2 /* Next */: {
for (let i = activeIndex + 1; i < items.length; ++i) {
if (!resolvers.resolveDisabled(items[i], i, items)) {
return i;
}
}
return currentActiveIndex;
}
case 3 /* Last */: {
for (let i = items.length - 1; i >= 0; --i) {
if (!resolvers.resolveDisabled(items[i], i, items)) {
return i;
}
}
return currentActiveIndex;
}
case 4 /* Specific */: {
for (let i = 0; i < items.length; ++i) {
if (resolvers.resolveId(items[i], i, items) === action.id) {
return i;
}
}
return currentActiveIndex;
}
case 5 /* Nothing */:
return null;
default:
assertNever(action);
}
}
// src/utils/form.ts
function objectToFormEntries(source = {}, parentKey = null, entries = []) {
for (let [key, value] of Object.entries(source)) {
append(entries, composeKey(parentKey, key), value);
}
return entries;
}
function composeKey(parent, key) {
return parent ? parent + "[" + key + "]" : key;
}
function append(entries, key, value) {
if (Array.isArray(value)) {
for (let [subkey, subvalue] of value.entries()) {
append(entries, composeKey(key, subkey.toString()), subvalue);
}
} else if (value instanceof Date) {
entries.push([key, value.toISOString()]);
} else if (typeof value === "boolean") {
entries.push([key, value ? "1" : "0"]);
} else if (typeof value === "string") {
entries.push([key, value]);
} else if (typeof value === "number") {
entries.push([key, `${value}`]);
} else if (value === null || value === void 0) {
entries.push([key, ""]);
} else {
objectToFormEntries(value, key, entries);
}
}
function attemptSubmit(elementInForm) {
var _a3, _b;
let form = (_a3 = elementInForm == null ? void 0 : elementInForm.form) != null ? _a3 : elementInForm.closest("form");
if (!form)
return;
for (let element of form.elements) {
if (element === elementInForm)
continue;
if (element.tagName === "INPUT" && element.type === "submit" || element.tagName === "BUTTON" && element.type === "submit" || element.nodeName === "INPUT" && element.type === "image") {
element.click();
return;
}
}
(_b = form.requestSubmit) == null ? void 0 : _b.call(form);
}
// src/components/combobox/combobox.tsx
function adjustOrderedState(state, adjustment = (i) => i) {
let currentActiveOption = state.activeOptionIndex !== null ? state.options[state.activeOptionIndex] : null;
let list = adjustment(state.options.slice());
let sortedOptions = list.length > 0 && list[0].dataRef.current.order !== null ? (
// Prefer sorting based on the `order`
list.sort((a, z) => a.dataRef.current.order - z.dataRef.current.order)
) : (
// Fallback to much slower DOM order
sortByDomNode(list, (option) => option.dataRef.current.domRef.current)
);
let adjustedActiveOptionIndex = currentActiveOption ? sortedOptions.indexOf(currentActiveOption) : null;
if (adjustedActiveOptionIndex === -1) {
adjustedActiveOptionIndex = null;
}
return {
options: sortedOptions,
activeOptionIndex: adjustedActiveOptionIndex
};
}
var reducers = {
[1 /* CloseCombobox */](state) {
var _a3;
if ((_a3 = state.dataRef.current) == null ? void 0 : _a3.disabled)
return state;
if (state.comboboxState === 1 /* Closed */)
return state;
return { ...state, activeOptionIndex: null, comboboxState: 1 /* Closed */ };
},
[0 /* OpenCombobox */](state) {
var _a3, _b;
if ((_a3 = state.dataRef.current) == null ? void 0 : _a3.disabled)
return state;
if (state.comboboxState === 0 /* Open */)
return state;
if ((_b = state.dataRef.current) == null ? void 0 : _b.value) {
let idx = state.dataRef.current.calculateIndex(state.dataRef.current.value);
if (idx !== -1) {
return {
...state,
activeOptionIndex: idx,
comboboxState: 0 /* Open */
};
}
}
return { ...state, comboboxState: 0 /* Open */ };
},
[2 /* GoToOption */](state, action) {
var _a3, _b, _c, _d, _e;
if ((_a3 = state.dataRef.current) == null ? void 0 : _a3.disabled)
return state;
if (((_b = state.dataRef.current) == null ? void 0 : _b.optionsRef.current) && !((_c = state.dataRef.current) == null ? void 0 : _c.optionsPropsRef.current.static) && state.comboboxState === 1 /* Closed */) {
return state;
}
if (state.virtual) {
let activeOptionIndex2 = action.focus === 4 /* Specific */ ? action.idx : calculateActiveIndex(action, {
resolveItems: () => state.virtual.options,
resolveActiveIndex: () => {
var _a4, _b2;
return (_b2 = (_a4 = state.activeOptionIndex) != null ? _a4 : state.virtual.options.findIndex((option) => !state.virtual.disabled(option))) != null ? _b2 : null;
},
resolveDisabled: state.virtual.disabled,
resolveId() {
throw new Error("Function not implemented.");
}
});
let activationTrigger2 = (_d = action.trigger) != null ? _d : 2 /* Other */;
if (state.activeOptionIndex === activeOptionIndex2 && state.activationTrigger === activationTrigger2) {
return state;
}
return {
...state,
activeOptionIndex: activeOptionIndex2,
activationTrigger: activationTrigger2
};
}
let adjustedState = adjustOrderedState(state);
if (adjustedState.activeOptionIndex === null) {
let localActiveOptionIndex = adjustedState.options.findIndex(
(option) => !option.dataRef.current.disabled
);
if (localActiveOptionIndex !== -1) {
adjustedState.activeOptionIndex = localActiveOptionIndex;
}
}
let activeOptionIndex = action.focus === 4 /* Specific */ ? action.idx : calculateActiveIndex(action, {
resolveItems: () => adjustedState.options,
resolveActiveIndex: () => adjustedState.activeOptionIndex,
resolveId: (item) => item.id,
resolveDisabled: (item) => item.dataRef.current.disabled
});
let activationTrigger = (_e = action.trigger) != null ? _e : 2 /* Other */;
if (state.activeOptionIndex === activeOptionIndex && state.activationTrigger === activationTrigger) {
return state;
}
return {
...state,
...adjustedState,
activeOptionIndex,
activationTrigger
};
},
[3 /* RegisterOption */]: (state, action) => {
var _a3, _b, _c;
if ((_a3 = state.dataRef.current) == null ? void 0 : _a3.virtual) {
return {
...state,
options: [...state.options, action.payload]
};
}
let option = action.payload;
let adjustedState = adjustOrderedState(state, (options) => {
options.push(option);
return options;
});
if (state.activeOptionIndex === null) {
if ((_b = state.dataRef.current) == null ? void 0 : _b.isSelected(action.payload.dataRef.current.value)) {
adjustedState.activeOptionIndex = adjustedState.options.indexOf(option);
}
}
let nextState = {
...state,
...adjustedState,
activationTrigger: 2 /* Other */
};
if (((_c = state.dataRef.current) == null ? void 0 : _c.__demoMode) && state.dataRef.current.value === void 0) {
nextState.activeOptionIndex = 0;
}
return nextState;
},
[4 /* UnregisterOption */]: (state, action) => {
var _a3;
if ((_a3 = state.dataRef.current) == null ? void 0 : _a3.virtual) {
return {
...state,
options: state.options.filter((option) => option.id !== action.id)
};
}
let adjustedState = adjustOrderedState(state, (options) => {
let idx = options.findIndex((option) => option.id === action.id);
if (idx !== -1)
options.splice(idx, 1);
return options;
});
return {
...state,
...adjustedState,
activationTrigger: 2 /* Other */
};
},
[5 /* RegisterLabel */]: (state, action) => {
if (state.labelId === action.id) {
return state;
}
return {
...state,
labelId: action.id
};
},
[6 /* SetActivationTrigger */]: (state, action) => {
if (state.activationTrigger === action.trigger) {
return state;
}
return {
...state,
activationTrigger: action.trigger
};
},
[7 /* UpdateVirtualOptions */]: (state, action) => {
var _a3;
if (((_a3 = state.virtual) == null ? void 0 : _a3.options) === action.options) {
return state;
}
let adjustedActiveOptionIndex = state.activeOptionIndex;
if (state.activeOptionIndex !== null) {
let idx = action.options.indexOf(state.virtual.options[state.activeOptionIndex]);
if (idx !== -1) {
adjustedActiveOptionIndex = idx;
} else {
adjustedActiveOptionIndex = null;
}
}
return {
...state,
activeOptionIndex: adjustedActiveOptionIndex,
virtual: Object.assign({}, state.virtual, { options: action.options })
};
}
};
var ComboboxActionsContext = (0, import_react19.createContext)(null);
ComboboxActionsContext.displayName = "ComboboxActionsContext";
function useActions(component) {
let context = (0, import_react19.useContext)(ComboboxActionsContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Combobox /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useActions);
throw err;
}
return context;
}
var VirtualContext = (0, import_react19.createContext)(null);
function VirtualProvider(props) {
var _a3;
let data = useData("VirtualProvider");
let [paddingStart, paddingEnd] = (0, import_react19.useMemo)(() => {
let el = data.optionsRef.current;
if (!el)
return [0, 0];
let styles = window.getComputedStyle(el);
return [
parseFloat(styles.paddingBlockStart || styles.paddingTop),
parseFloat(styles.paddingBlockEnd || styles.paddingBottom)
];
}, [data.optionsRef.current]);
let virtualizer = useVirtualizer({
scrollPaddingStart: paddingStart,
scrollPaddingEnd: paddingEnd,
count: data.virtual.options.length,
estimateSize() {
return 40;
},
getScrollElement() {
var _a4;
return (_a4 = data.optionsRef.current) != null ? _a4 : null;
},
overscan: 12
});
let [baseKey, setBaseKey] = (0, import_react19.useState)(0);
useIsoMorphicEffect(() => {
setBaseKey((v) => v + 1);
}, [(_a3 = data.virtual) == null ? void 0 : _a3.options]);
return /* @__PURE__ */ import_react19.default.createElement(VirtualContext.Provider, { value: virtualizer }, /* @__PURE__ */ import_react19.default.createElement(
"div",
{
style: {
position: "relative",
width: "100%",
height: `${virtualizer.getTotalSize()}px`
},
ref: (el) => {
if (!el) {
return;
}
{
if (typeof process !== "undefined" && process.env.JEST_WORKER_ID !== void 0) {
return;
}
if (data.activationTrigger === 0 /* Pointer */) {
return;
}
if (data.activeOptionIndex !== null && data.virtual.options.length > data.activeOptionIndex) {
virtualizer.scrollToIndex(data.activeOptionIndex);
}
}
}
},
virtualizer.getVirtualItems().map((item) => {
var _a4;
return /* @__PURE__ */ import_react19.default.createElement(import_react19.Fragment, { key: item.key }, import_react19.default.cloneElement(
(_a4 = props.children) == null ? void 0 : _a4.call(props, {
option: data.virtual.options[item.index],
open: data.comboboxState === 0 /* Open */
}),
{
key: `${baseKey}-${item.key}`,
"data-index": item.index,
"aria-setsize": data.virtual.options.length,
"aria-posinset": item.index + 1,
style: {
position: "absolute",
top: 0,
left: 0,
transform: `translateY(${item.start}px)`,
overflowAnchor: "none"
}
}
));
})
));
}
var ComboboxDataContext = (0, import_react19.createContext)(null);
ComboboxDataContext.displayName = "ComboboxDataContext";
function useData(component) {
let context = (0, import_react19.useContext)(ComboboxDataContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Combobox /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useData);
throw err;
}
return context;
}
function stateReducer(state, action) {
return match(action.type, reducers, state, action);
}
var DEFAULT_COMBOBOX_TAG = import_react19.Fragment;
function ComboboxFn(props, ref) {
var _a3;
let {
value: controlledValue,
defaultValue,
onChange: controlledOnChange,
form: formName,
name,
by = null,
disabled = false,
__demoMode = false,
nullable = false,
multiple = false,
// @ts-expect-error
immediate: _immediate = false,
// @ts-expect-error
virtual: _virtual = null,
...theirProps
} = props;
let immediate = false;
let virtual = null;
let [value = multiple ? [] : void 0, theirOnChange] = useControllable(
controlledValue,
controlledOnChange,
defaultValue
);
let [state, dispatch] = (0, import_react19.useReducer)(stateReducer, {
dataRef: (0, import_react19.createRef)(),
comboboxState: __demoMode ? 0 /* Open */ : 1 /* Closed */,
options: [],
virtual: virtual ? { options: virtual.options, disabled: (_a3 = virtual.disabled) != null ? _a3 : () => false } : null,
activeOptionIndex: null,
activationTrigger: 2 /* Other */,
labelId: null
});
let defaultToFirstOption = (0, import_react19.useRef)(false);
let optionsPropsRef = (0, import_react19.useRef)({ static: false, hold: false });
let labelRef = (0, import_react19.useRef)(null);
let inputRef = (0, import_react19.useRef)(null);
let buttonRef = (0, import_react19.useRef)(null);
let optionsRef = (0, import_react19.useRef)(null);
let compare = useEvent(
// @ts-expect-error Eventually we'll want to tackle this, but for now this will do.
typeof by === "string" ? (a, z) => {
let property = by;
return (a == null ? void 0 : a[property]) === (z == null ? void 0 : z[property]);
} : by != null ? by : (a, z) => a === z
);
let calculateIndex = useEvent((value2) => {
if (virtual) {
if (by === null) {
return virtual.options.indexOf(value2);
} else {
return virtual.options.findIndex((other) => compare(other, value2));
}
} else {
return state.options.findIndex((other) => compare(other.dataRef.current.value, value2));
}
});
let isSelected = (0, import_react19.useCallback)(
(other) => match(data.mode, {
[1 /* Multi */]: () => value.some((option) => compare(option, other)),
[0 /* Single */]: () => compare(value, other)
}),
[value]
);
let isActive = useEvent((other) => {
return state.activeOptionIndex === calculateIndex(other);
});
let data = (0, import_react19.useMemo)(
() => ({
...state,
immediate,
optionsPropsRef,
labelRef,
inputRef,
buttonRef,
optionsRef,
value,
defaultValue,
disabled,
mode: multiple ? 1 /* Multi */ : 0 /* Single */,
virtual: state.virtual,
get activeOptionIndex() {
if (defaultToFirstOption.current && state.activeOptionIndex === null && (virtual ? virtual.options.length > 0 : state.options.length > 0)) {
if (virtual) {
let localActiveOptionIndex2 = virtual.options.findIndex(
(option) => {
var _a4, _b;
return !((_b = (_a4 = virtual == null ? void 0 : virtual.disabled) == null ? void 0 : _a4.call(virtual, option)) != null ? _b : false);
}
);
if (localActiveOptionIndex2 !== -1) {
return localActiveOptionIndex2;
}
}
let localActiveOptionIndex = state.options.findIndex((option) => {
return !option.dataRef.current.disabled;
});
if (localActiveOptionIndex !== -1) {
return localActiveOptionIndex;
}
}
return state.activeOptionIndex;
},
calculateIndex,
compare,
isSelected,
isActive,
nullable,
__demoMode
}),
[value, defaultValue, disabled, multiple, nullable, __demoMode, state, virtual]
);
useIsoMorphicEffect(() => {
if (!virtual)
return;
dispatch({ type: 7 /* UpdateVirtualOptions */, options: virtual.options });
}, [virtual, virtual == null ? void 0 : virtual.options]);
useIsoMorphicEffect(() => {
state.dataRef.current = data;
}, [data]);
useOutsideClick(
[data.buttonRef, data.inputRef, data.optionsRef],
() => actions.closeCombobox(),
data.comboboxState === 0 /* Open */
);
let slot = (0, import_react19.useMemo)(
() => {
var _a4, _b, _c;
return {
open: data.comboboxState === 0 /* Open */,
disabled,
activeIndex: data.activeOptionIndex,
activeOption: data.activeOptionIndex === null ? null : data.virtual ? data.virtual.options[(_a4 = data.activeOptionIndex) != null ? _a4 : 0] : (_c = (_b = data.options[data.activeOptionIndex]) == null ? void 0 : _b.dataRef.current.value) != null ? _c : null,
value
};
},
[data, disabled, value]
);
let selectActiveOption = useEvent(() => {
if (data.activeOptionIndex === null)
return;
if (data.virtual) {
onChange(data.virtual.options[data.activeOptionIndex]);
} else {
let { dataRef } = data.options[data.activeOptionIndex];
onChange(dataRef.current.value);
}
actions.goToOption(4 /* Specific */, data.activeOptionIndex);
});
let openCombobox = useEvent(() => {
dispatch({ type: 0 /* OpenCombobox */ });
defaultToFirstOption.current = true;
});
let closeCombobox = useEvent(() => {
dispatch({ type: 1 /* CloseCombobox */ });
defaultToFirstOption.current = false;
});
let goToOption = useEvent((focus, idx, trigger) => {
defaultToFirstOption.current = false;
if (focus === 4 /* Specific */) {
return dispatch({ type: 2 /* GoToOption */, focus: 4 /* Specific */, idx, trigger });
}
return dispatch({ type: 2 /* GoToOption */, focus, trigger });
});
let registerOption = useEvent((id, dataRef) => {
dispatch({ type: 3 /* RegisterOption */, payload: { id, dataRef } });
return () => {
if (data.isActive(dataRef.current.value)) {
defaultToFirstOption.current = true;
}
dispatch({ type: 4 /* UnregisterOption */, id });
};
});
let registerLabel = useEvent((id) => {
dispatch({ type: 5 /* RegisterLabel */, id });
return () => dispatch({ type: 5 /* RegisterLabel */, id: null });
});
let onChange = useEvent((value2) => {
return match(data.mode, {
[0 /* Single */]() {
return theirOnChange == null ? void 0 : theirOnChange(value2);
},
[1 /* Multi */]() {
let copy = data.value.slice();
let idx = copy.findIndex((item) => compare(item, value2));
if (idx === -1) {
copy.push(value2);
} else {
copy.splice(idx, 1);
}
return theirOnChange == null ? void 0 : theirOnChange(copy);
}
});
});
let setActivationTrigger = useEvent((trigger) => {
dispatch({ type: 6 /* SetActivationTrigger */, trigger });
});
let actions = (0, import_react19.useMemo)(
() => ({
onChange,
registerOption,
registerLabel,
goToOption,
closeCombobox,
openCombobox,
setActivationTrigger,
selectActiveOption
}),
[]
);
let ourProps = ref === null ? {} : { ref };
let form = (0, import_react19.useRef)(null);
let d = useDisposables();
(0, import_react19.useEffect)(() => {
if (!form.current)
return;
if (defaultValue === void 0)
return;
d.addEventListener(form.current, "reset", () => {
theirOnChange == null ? void 0 : theirOnChange(defaultValue);
});
}, [
form,
theirOnChange
/* Explicitly ignoring `defaultValue` */
]);
return /* @__PURE__ */ import_react19.default.createElement(ComboboxActionsContext.Provider, { value: actions }, /* @__PURE__ */ import_react19.default.createElement(ComboboxDataContext.Provider, { value: data }, /* @__PURE__ */ import_react19.default.createElement(
OpenClosedProvider,
{
value: match(data.comboboxState, {
[0 /* Open */]: 1 /* Open */,
[1 /* Closed */]: 2 /* Closed */
})
},
name != null && value != null && objectToFormEntries({ [name]: value }).map(([name2, value2], idx) => /* @__PURE__ */ import_react19.default.createElement(
Hidden,
{
features: 4 /* Hidden */,
ref: idx === 0 ? (element) => {
var _a4;
form.current = (_a4 = element == null ? void 0 : element.closest("form")) != null ? _a4 : null;
} : void 0,
...compact({
key: name2,
as: "input",
type: "hidden",
hidden: true,
readOnly: true,
form: formName,
disabled,
name: name2,
value: value2
})
}
)),
render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_COMBOBOX_TAG,
name: "Combobox"
})
)));
}
var DEFAULT_INPUT_TAG = "input";
function InputFn(props, ref) {
var _a3, _b, _c, _d, _e;
let internalId = useId();
let {
id = `headlessui-combobox-input-${internalId}`,
onChange,
displayValue,
// @ts-ignore: We know this MAY NOT exist for a given tag but we only care when it _does_ exist.
type = "text",
...theirProps
} = props;
let data = useData("Combobox.Input");
let actions = useActions("Combobox.Input");
let inputRef = useSyncRefs(data.inputRef, ref);
let ownerDocument = useOwnerDocument(data.inputRef);
let isTyping = (0, import_react19.useRef)(false);
let d = useDisposables();
let clear = useEvent(() => {
actions.onChange(null);
if (data.optionsRef.current) {
data.optionsRef.current.scrollTop = 0;
}
actions.goToOption(5 /* Nothing */);
});
let currentDisplayValue = function() {
var _a4;
if (typeof displayValue === "function" && data.value !== void 0) {
return (_a4 = displayValue(data.value)) != null ? _a4 : "";
} else if (typeof data.value === "string") {
return data.value;
} else {
return "";
}
}();
useWatch(
([currentDisplayValue2, state], [oldCurrentDisplayValue, oldState]) => {
if (isTyping.current)
return;
let input = data.inputRef.current;
if (!input)
return;
if (oldState === 0 /* Open */ && state === 1 /* Closed */) {
input.value = currentDisplayValue2;
} else if (currentDisplayValue2 !== oldCurrentDisplayValue) {
input.value = currentDisplayValue2;
}
requestAnimationFrame(() => {
if (isTyping.current)
return;
if (!input)
return;
if ((ownerDocument == null ? void 0 : ownerDocument.activeElement) !== input)
return;
let { selectionStart, selectionEnd } = input;
if (Math.abs((selectionEnd != null ? selectionEnd : 0) - (selectionStart != null ? selectionStart : 0)) !== 0)
return;
if (selectionStart !== 0)
return;
input.setSelectionRange(input.value.length, input.value.length);
});
},
[currentDisplayValue, data.comboboxState, ownerDocument]
);
useWatch(
([newState], [oldState]) => {
if (newState === 0 /* Open */ && oldState === 1 /* Closed */) {
if (isTyping.current)
return;
let input = data.inputRef.current;
if (!input)
return;
let currentValue = input.value;
let { selectionStart, selectionEnd, selectionDirection } = input;
input.value = "";
input.value = currentValue;
if (selectionDirection !== null) {
input.setSelectionRange(selectionStart, selectionEnd, selectionDirection);
} else {
input.setSelectionRange(selectionStart, selectionEnd);
}
}
},
[data.comboboxState]
);
let isComposing = (0, import_react19.useRef)(false);
let handleCompositionStart = useEvent(() => {
isComposing.current = true;
});
let handleCompositionEnd = useEvent(() => {
d.nextFrame(() => {
isComposing.current = false;
});
});
let handleKeyDown = useEvent((event) => {
isTyping.current = true;
switch (event.key) {
case "Enter" /* Enter */:
isTyping.current = false;
if (data.comboboxState !== 0 /* Open */)
return;
if (isComposing.current)
return;
event.preventDefault();
event.stopPropagation();
if (data.activeOptionIndex === null) {
actions.closeCombobox();
return;
}
actions.selectActiveOption();
if (data.mode === 0 /* Single */) {
actions.closeCombobox();
}
break;
case "ArrowDown" /* ArrowDown */:
isTyping.current = false;
event.preventDefault();
event.stopPropagation();
return match(data.comboboxState, {
[0 /* Open */]: () => actions.goToOption(2 /* Next */),
[1 /* Closed */]: () => actions.openCombobox()
});
case "ArrowUp" /* ArrowUp */:
isTyping.current = false;
event.preventDefault();
event.stopPropagation();
return match(data.comboboxState, {
[0 /* Open */]: () => actions.goToOption(1 /* Previous */),
[1 /* Closed */]: () => {
actions.openCombobox();
d.nextFrame(() => {
if (!data.value) {
actions.goToOption(3 /* Last */);
}
});
}
});
case "Home" /* Home */:
if (event.shiftKey) {
break;
}
isTyping.current = false;
event.preventDefault();
event.stopPropagation();
return actions.goToOption(0 /* First */);
case "PageUp" /* PageUp */:
isTyping.current = false;
event.preventDefault();
event.stopPropagation();
return actions.goToOption(0 /* First */);
case "End" /* End */:
if (event.shiftKey) {
break;
}
isTyping.current = false;
event.preventDefault();
event.stopPropagation();
return actions.goToOption(3 /* Last */);
case "PageDown" /* PageDown */:
isTyping.current = false;
event.preventDefault();
event.stopPropagation();
return actions.goToOption(3 /* Last */);
case "Escape" /* Escape */:
isTyping.current = false;
if (data.comboboxState !== 0 /* Open */)
return;
event.preventDefault();
if (data.optionsRef.current && !data.optionsPropsRef.current.static) {
event.stopPropagation();
}
if (data.nullable && data.mode === 0 /* Single */) {
if (data.value === null) {
clear();
}
}
return actions.closeCombobox();
case "Tab" /* Tab */:
isTyping.current = false;
if (data.comboboxState !== 0 /* Open */)
return;
if (data.mode === 0 /* Single */ && data.activationTrigger !== 1 /* Focus */) {
actions.selectActiveOption();
}
actions.closeCombobox();
break;
}
});
let handleChange = useEvent((event) => {
onChange == null ? void 0 : onChange(event);
if (data.nullable && data.mode === 0 /* Single */) {
if (event.target.value === "") {
clear();
}
}
actions.openCombobox();
});
let handleBlur = useEvent((event) => {
var _a4, _b2, _c2;
let relatedTarget = (_a4 = event.relatedTarget) != null ? _a4 : history.find((x) => x !== event.currentTarget);
isTyping.current = false;
if ((_b2 = data.optionsRef.current) == null ? void 0 : _b2.contains(relatedTarget)) {
return;
}
if ((_c2 = data.buttonRef.current) == null ? void 0 : _c2.contains(relatedTarget)) {
return;
}
if (data.comboboxState !== 0 /* Open */)
return;
event.preventDefault();
if (data.mode === 0 /* Single */) {
if (data.nullable && data.value === null) {
clear();
} else if (data.activationTrigger !== 1 /* Focus */) {
actions.selectActiveOption();
}
}
return actions.closeCombobox();
});
let handleFocus = useEvent((event) => {
var _a4, _b2, _c2;
let relatedTarget = (_a4 = event.relatedTarget) != null ? _a4 : history.find((x) => x !== event.currentTarget);
if ((_b2 = data.buttonRef.current) == null ? void 0 : _b2.contains(relatedTarget))
return;
if ((_c2 = data.optionsRef.current) == null ? void 0 : _c2.contains(relatedTarget))
return;
if (data.disabled)
return;
if (!data.immediate)
return;
if (data.comboboxState === 0 /* Open */)
return;
actions.openCombobox();
d.nextFrame(() => {
actions.setActivationTrigger(1 /* Focus */);
});
});
let labelledby = useComputed(() => {
if (!data.labelId)
return void 0;
return [data.labelId].join(" ");
}, [data.labelId]);
let slot = (0, import_react19.useMemo)(
() => ({ open: data.comboboxState === 0 /* Open */, disabled: data.disabled }),
[data]
);
let ourProps = {
ref: inputRef,
id,
role: "combobox",
type,
"aria-controls": (_a3 = data.optionsRef.current) == null ? void 0 : _a3.id,
"aria-expanded": data.comboboxState === 0 /* Open */,
"aria-activedescendant": data.activeOptionIndex === null ? void 0 : data.virtual ? (_b = data.options.find(
(option) => {
var _a4;
return !((_a4 = data.virtual) == null ? void 0 : _a4.disabled(option.dataRef.current.value)) && data.compare(
option.dataRef.current.value,
data.virtual.options[data.activeOptionIndex]
);
}
)) == null ? void 0 : _b.id : (_c = data.options[data.activeOptionIndex]) == null ? void 0 : _c.id,
"aria-labelledby": labelledby,
"aria-autocomplete": "list",
defaultValue: (_e = (_d = props.defaultValue) != null ? _d : data.defaultValue !== void 0 ? displayValue == null ? void 0 : displayValue(data.defaultValue) : null) != null ? _e : data.defaultValue,
disabled: data.disabled,
onCompositionStart: handleCompositionStart,
onCompositionEnd: handleCompositionEnd,
onKeyDown: handleKeyDown,
onChange: handleChange,
onFocus: handleFocus,
onBlur: handleBlur
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_INPUT_TAG,
name: "Combobox.Input"
});
}
var DEFAULT_BUTTON_TAG = "button";
function ButtonFn(props, ref) {
var _a3;
let data = useData("Combobox.Button");
let actions = useActions("Combobox.Button");
let buttonRef = useSyncRefs(data.buttonRef, ref);
let internalId = useId();
let { id = `headlessui-combobox-button-${internalId}`, ...theirProps } = props;
let d = useDisposables();
let handleKeyDown = useEvent((event) => {
switch (event.key) {
case "ArrowDown" /* ArrowDown */:
event.preventDefault();
event.stopPropagation();
if (data.comboboxState === 1 /* Closed */) {
actions.openCombobox();
}
return d.nextFrame(() => {
var _a4;
return (_a4 = data.inputRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
case "ArrowUp" /* ArrowUp */:
event.preventDefault();
event.stopPropagation();
if (data.comboboxState === 1 /* Closed */) {
actions.openCombobox();
d.nextFrame(() => {
if (!data.value) {
actions.goToOption(3 /* Last */);
}
});
}
return d.nextFrame(() => {
var _a4;
return (_a4 = data.inputRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
case "Escape" /* Escape */:
if (data.comboboxState !== 0 /* Open */)
return;
event.preventDefault();
if (data.optionsRef.current && !data.optionsPropsRef.current.static) {
event.stopPropagation();
}
actions.closeCombobox();
return d.nextFrame(() => {
var _a4;
return (_a4 = data.inputRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
default:
return;
}
});
let handleClick = useEvent((event) => {
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
if (data.comboboxState === 0 /* Open */) {
actions.closeCombobox();
} else {
event.preventDefault();
actions.openCombobox();
}
d.nextFrame(() => {
var _a4;
return (_a4 = data.inputRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
});
let labelledby = useComputed(() => {
if (!data.labelId)
return void 0;
return [data.labelId, id].join(" ");
}, [data.labelId, id]);
let slot = (0, import_react19.useMemo)(
() => ({
open: data.comboboxState === 0 /* Open */,
disabled: data.disabled,
value: data.value
}),
[data]
);
let ourProps = {
ref: buttonRef,
id,
type: useResolveButtonType(props, data.buttonRef),
tabIndex: -1,
"aria-haspopup": "listbox",
"aria-controls": (_a3 = data.optionsRef.current) == null ? void 0 : _a3.id,
"aria-expanded": data.comboboxState === 0 /* Open */,
"aria-labelledby": labelledby,
disabled: data.disabled,
onClick: handleClick,
onKeyDown: handleKeyDown
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_BUTTON_TAG,
name: "Combobox.Button"
});
}
var DEFAULT_LABEL_TAG = "label";
function LabelFn(props, ref) {
let internalId = useId();
let { id = `headlessui-combobox-label-${internalId}`, ...theirProps } = props;
let data = useData("Combobox.Label");
let actions = useActions("Combobox.Label");
let labelRef = useSyncRefs(data.labelRef, ref);
useIsoMorphicEffect(() => actions.registerLabel(id), [id]);
let handleClick = useEvent(() => {
var _a3;
return (_a3 = data.inputRef.current) == null ? void 0 : _a3.focus({ preventScroll: true });
});
let slot = (0, import_react19.useMemo)(
() => ({ open: data.comboboxState === 0 /* Open */, disabled: data.disabled }),
[data]
);
let ourProps = { ref: labelRef, id, onClick: handleClick };
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_LABEL_TAG,
name: "Combobox.Label"
});
}
var DEFAULT_OPTIONS_TAG = "ul";
var OptionsRenderFeatures = 1 /* RenderStrategy */ | 2 /* Static */;
function OptionsFn(props, ref) {
let internalId = useId();
let { id = `headlessui-combobox-options-${internalId}`, hold = false, ...theirProps } = props;
let data = useData("Combobox.Options");
let optionsRef = useSyncRefs(data.optionsRef, ref);
let usesOpenClosedState = useOpenClosed();
let visible = (() => {
if (usesOpenClosedState !== null) {
return (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
return data.comboboxState === 0 /* Open */;
})();
useIsoMorphicEffect(() => {
var _a3;
data.optionsPropsRef.current.static = (_a3 = props.static) != null ? _a3 : false;
}, [data.optionsPropsRef, props.static]);
useIsoMorphicEffect(() => {
data.optionsPropsRef.current.hold = hold;
}, [data.optionsPropsRef, hold]);
useTreeWalker({
container: data.optionsRef.current,
enabled: data.comboboxState === 0 /* Open */,
accept(node) {
if (node.getAttribute("role") === "option")
return NodeFilter.FILTER_REJECT;
if (node.hasAttribute("role"))
return NodeFilter.FILTER_SKIP;
return NodeFilter.FILTER_ACCEPT;
},
walk(node) {
node.setAttribute("role", "none");
}
});
let labelledby = useComputed(
() => {
var _a3, _b;
return (_b = data.labelId) != null ? _b : (_a3 = data.buttonRef.current) == null ? void 0 : _a3.id;
},
[data.labelId, data.buttonRef.current]
);
let slot = (0, import_react19.useMemo)(
() => ({ open: data.comboboxState === 0 /* Open */, option: void 0 }),
[data]
);
let ourProps = {
"aria-labelledby": labelledby,
role: "listbox",
"aria-multiselectable": data.mode === 1 /* Multi */ ? true : void 0,
id,
ref: optionsRef
};
if (data.virtual && data.comboboxState === 0 /* Open */) {
Object.assign(theirProps, {
// @ts-expect-error The `children` prop now is a callback function that receives `{ option }`.
children: /* @__PURE__ */ import_react19.default.createElement(VirtualProvider, null, theirProps.children)
});
}
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OPTIONS_TAG,
features: OptionsRenderFeatures,
visible,
name: "Combobox.Options"
});
}
var DEFAULT_OPTION_TAG = "li";
function OptionFn(props, ref) {
var _a3;
let internalId = useId();
let {
id = `headlessui-combobox-option-${internalId}`,
disabled = false,
value,
order = null,
...theirProps
} = props;
let data = useData("Combobox.Option");
let actions = useActions("Combobox.Option");
let active = data.virtual ? data.activeOptionIndex === data.calculateIndex(value) : data.activeOptionIndex === null ? false : ((_a3 = data.options[data.activeOptionIndex]) == null ? void 0 : _a3.id) === id;
let selected = data.isSelected(value);
let internalOptionRef = (0, import_react19.useRef)(null);
let bag = useLatestValue({
disabled,
value,
domRef: internalOptionRef,
order
});
let virtualizer = (0, import_react19.useContext)(VirtualContext);
let optionRef = useSyncRefs(
ref,
internalOptionRef,
virtualizer ? virtualizer.measureElement : null
);
let select = useEvent(() => actions.onChange(value));
useIsoMorphicEffect(() => actions.registerOption(id, bag), [bag, id]);
let enableScrollIntoView = (0, import_react19.useRef)(data.virtual || data.__demoMode ? false : true);
useIsoMorphicEffect(() => {
if (!data.virtual)
return;
if (!data.__demoMode)
return;
let d = disposables();
d.requestAnimationFrame(() => {
enableScrollIntoView.current = true;
});
return d.dispose;
}, [data.virtual, data.__demoMode]);
useIsoMorphicEffect(() => {
if (!enableScrollIntoView.current)
return;
if (data.comboboxState !== 0 /* Open */)
return;
if (!active)
return;
if (data.activationTrigger === 0 /* Pointer */)
return;
let d = disposables();
d.requestAnimationFrame(() => {
var _a4, _b;
(_b = (_a4 = internalOptionRef.current) == null ? void 0 : _a4.scrollIntoView) == null ? void 0 : _b.call(_a4, { block: "nearest" });
});
return d.dispose;
}, [
internalOptionRef,
active,
data.comboboxState,
data.activationTrigger,
/* We also want to trigger this when the position of the active item changes so that we can re-trigger the scrollIntoView */
data.activeOptionIndex
]);
let handleClick = useEvent((event) => {
var _a4;
if (disabled || ((_a4 = data.virtual) == null ? void 0 : _a4.disabled(value)))
return event.preventDefault();
select();
if (!isMobile()) {
requestAnimationFrame(() => {
var _a5;
return (_a5 = data.inputRef.current) == null ? void 0 : _a5.focus({ preventScroll: true });
});
}
if (data.mode === 0 /* Single */) {
requestAnimationFrame(() => actions.closeCombobox());
}
});
let handleFocus = useEvent(() => {
var _a4;
if (disabled || ((_a4 = data.virtual) == null ? void 0 : _a4.disabled(value))) {
return actions.goToOption(5 /* Nothing */);
}
let idx = data.calculateIndex(value);
actions.goToOption(4 /* Specific */, idx);
});
let pointer = useTrackedPointer();
let handleEnter = useEvent((evt) => pointer.update(evt));
let handleMove = useEvent((evt) => {
var _a4;
if (!pointer.wasMoved(evt))
return;
if (disabled || ((_a4 = data.virtual) == null ? void 0 : _a4.disabled(value)))
return;
if (active)
return;
let idx = data.calculateIndex(value);
actions.goToOption(4 /* Specific */, idx, 0 /* Pointer */);
});
let handleLeave = useEvent((evt) => {
var _a4;
if (!pointer.wasMoved(evt))
return;
if (disabled || ((_a4 = data.virtual) == null ? void 0 : _a4.disabled(value)))
return;
if (!active)
return;
if (data.optionsPropsRef.current.hold)
return;
actions.goToOption(5 /* Nothing */);
});
let slot = (0, import_react19.useMemo)(
() => ({ active, selected, disabled }),
[active, selected, disabled]
);
let ourProps = {
id,
ref: optionRef,
role: "option",
tabIndex: disabled === true ? void 0 : -1,
"aria-disabled": disabled === true ? true : void 0,
// According to the WAI-ARIA best practices, we should use aria-checked for
// multi-select,but Voice-Over disagrees. So we use aria-checked instead for
// both single and multi-select.
"aria-selected": selected,
disabled: void 0,
// Never forward the `disabled` prop
onClick: handleClick,
onFocus: handleFocus,
onPointerEnter: handleEnter,
onMouseEnter: handleEnter,
onPointerMove: handleMove,
onMouseMove: handleMove,
onPointerLeave: handleLeave,
onMouseLeave: handleLeave
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OPTION_TAG,
name: "Combobox.Option"
});
}
var ComboboxRoot = forwardRefWithAs(ComboboxFn);
var Button = forwardRefWithAs(ButtonFn);
var Input = forwardRefWithAs(InputFn);
var Label = forwardRefWithAs(LabelFn);
var Options = forwardRefWithAs(OptionsFn);
var Option = forwardRefWithAs(OptionFn);
var Combobox = Object.assign(ComboboxRoot, { Input, Button, Label, Options, Option });
// src/components/dialog/dialog.tsx
var import_react30 = __toESM(require("react"), 1);
// src/components/focus-trap/focus-trap.tsx
var import_react24 = __toESM(require("react"), 1);
// src/hooks/use-event-listener.ts
var import_react20 = require("react");
function useEventListener(element, type, listener, options) {
let listenerRef = useLatestValue(listener);
(0, import_react20.useEffect)(() => {
element = element != null ? element : window;
function handler(event) {
listenerRef.current(event);
}
element.addEventListener(type, handler, options);
return () => element.removeEventListener(type, handler, options);
}, [element, type, options]);
}
// src/hooks/use-is-mounted.ts
var import_react21 = require("react");
function useIsMounted() {
let mounted = (0, import_react21.useRef)(false);
useIsoMorphicEffect(() => {
mounted.current = true;
return () => {
mounted.current = false;
};
}, []);
return mounted;
}
// src/hooks/use-on-unmount.ts
var import_react22 = require("react");
function useOnUnmount(cb) {
let stableCb = useEvent(cb);
let trulyUnmounted = (0, import_react22.useRef)(false);
(0, import_react22.useEffect)(() => {
trulyUnmounted.current = false;
return () => {
trulyUnmounted.current = true;
microTask(() => {
if (!trulyUnmounted.current)
return;
stableCb();
});
};
}, [stableCb]);
}
// src/hooks/use-tab-direction.ts
var import_react23 = require("react");
function useTabDirection() {
let direction = (0, import_react23.useRef)(0 /* Forwards */);
useWindowEvent(
"keydown",
(event) => {
if (event.key === "Tab") {
direction.current = event.shiftKey ? 1 /* Backwards */ : 0 /* Forwards */;
}
},
true
);
return direction;
}
// src/components/focus-trap/focus-trap.tsx
function resolveContainers(containers) {
if (!containers)
return /* @__PURE__ */ new Set();
if (typeof containers === "function")
return new Set(containers());
let all = /* @__PURE__ */ new Set();
for (let container of containers.current) {
if (container.current instanceof HTMLElement) {
all.add(container.current);
}
}
return all;
}
var DEFAULT_FOCUS_TRAP_TAG = "div";
var Features3 = /* @__PURE__ */ ((Features4) => {
Features4[Features4["None"] = 1] = "None";
Features4[Features4["InitialFocus"] = 2] = "InitialFocus";
Features4[Features4["TabLock"] = 4] = "TabLock";
Features4[Features4["FocusLock"] = 8] = "FocusLock";
Features4[Features4["RestoreFocus"] = 16] = "RestoreFocus";
Features4[Features4["All"] = 30] = "All";
return Features4;
})(Features3 || {});
function FocusTrapFn(props, ref) {
let container = (0, import_react24.useRef)(null);
let focusTrapRef = useSyncRefs(container, ref);
let { initialFocus, containers, features = 30 /* All */, ...theirProps } = props;
if (!useServerHandoffComplete()) {
features = 1 /* None */;
}
let ownerDocument = useOwnerDocument(container);
useRestoreFocus({ ownerDocument }, Boolean(features & 16 /* RestoreFocus */));
let previousActiveElement = useInitialFocus(
{ ownerDocument, container, initialFocus },
Boolean(features & 2 /* InitialFocus */)
);
useFocusLock(
{ ownerDocument, container, containers, previousActiveElement },
Boolean(features & 8 /* FocusLock */)
);
let direction = useTabDirection();
let handleFocus = useEvent((e) => {
let el = container.current;
if (!el)
return;
let wrapper = false ? microTask : (cb) => cb();
wrapper(() => {
match(direction.current, {
[0 /* Forwards */]: () => {
focusIn(el, 1 /* First */, { skipElements: [e.relatedTarget] });
},
[1 /* Backwards */]: () => {
focusIn(el, 8 /* Last */, { skipElements: [e.relatedTarget] });
}
});
});
});
let d = useDisposables();
let recentlyUsedTabKey = (0, import_react24.useRef)(false);
let ourProps = {
ref: focusTrapRef,
onKeyDown(e) {
if (e.key == "Tab") {
recentlyUsedTabKey.current = true;
d.requestAnimationFrame(() => {
recentlyUsedTabKey.current = false;
});
}
},
onBlur(e) {
let allContainers = resolveContainers(containers);
if (container.current instanceof HTMLElement)
allContainers.add(container.current);
let relatedTarget = e.relatedTarget;
if (!(relatedTarget instanceof HTMLElement))
return;
if (relatedTarget.dataset.headlessuiFocusGuard === "true") {
return;
}
if (!contains(allContainers, relatedTarget)) {
if (recentlyUsedTabKey.current) {
focusIn(
container.current,
match(direction.current, {
[0 /* Forwards */]: () => 4 /* Next */,
[1 /* Backwards */]: () => 2 /* Previous */
}) | 16 /* WrapAround */,
{ relativeTo: e.target }
);
} else if (e.target instanceof HTMLElement) {
focusElement(e.target);
}
}
}
};
return /* @__PURE__ */ import_react24.default.createElement(import_react24.default.Fragment, null, Boolean(features & 4 /* TabLock */) && /* @__PURE__ */ import_react24.default.createElement(
Hidden,
{
as: "button",
type: "button",
"data-headlessui-focus-guard": true,
onFocus: handleFocus,
features: 2 /* Focusable */
}
), render({
ourProps,
theirProps,
defaultTag: DEFAULT_FOCUS_TRAP_TAG,
name: "FocusTrap"
}), Boolean(features & 4 /* TabLock */) && /* @__PURE__ */ import_react24.default.createElement(
Hidden,
{
as: "button",
type: "button",
"data-headlessui-focus-guard": true,
onFocus: handleFocus,
features: 2 /* Focusable */
}
));
}
var FocusTrapRoot = forwardRefWithAs(FocusTrapFn);
var FocusTrap = Object.assign(FocusTrapRoot, {
features: Features3
});
function useRestoreElement(enabled = true) {
let localHistory = (0, import_react24.useRef)(history.slice());
useWatch(
([newEnabled], [oldEnabled]) => {
if (oldEnabled === true && newEnabled === false) {
microTask(() => {
localHistory.current.splice(0);
});
}
if (oldEnabled === false && newEnabled === true) {
localHistory.current = history.slice();
}
},
[enabled, history, localHistory]
);
return useEvent(() => {
var _a3;
return (_a3 = localHistory.current.find((x) => x != null && x.isConnected)) != null ? _a3 : null;
});
}
function useRestoreFocus({ ownerDocument }, enabled) {
let getRestoreElement = useRestoreElement(enabled);
useWatch(() => {
if (enabled)
return;
if ((ownerDocument == null ? void 0 : ownerDocument.activeElement) === (ownerDocument == null ? void 0 : ownerDocument.body)) {
focusElement(getRestoreElement());
}
}, [enabled]);
useOnUnmount(() => {
if (!enabled)
return;
focusElement(getRestoreElement());
});
}
function useInitialFocus({
ownerDocument,
container,
initialFocus
}, enabled) {
let previousActiveElement = (0, import_react24.useRef)(null);
let mounted = useIsMounted();
useWatch(() => {
if (!enabled)
return;
let containerElement = container.current;
if (!containerElement)
return;
microTask(() => {
if (!mounted.current) {
return;
}
let activeElement = ownerDocument == null ? void 0 : ownerDocument.activeElement;
if (initialFocus == null ? void 0 : initialFocus.current) {
if ((initialFocus == null ? void 0 : initialFocus.current) === activeElement) {
previousActiveElement.current = activeElement;
return;
}
} else if (containerElement.contains(activeElement)) {
previousActiveElement.current = activeElement;
return;
}
if (initialFocus == null ? void 0 : initialFocus.current) {
focusElement(initialFocus.current);
} else {
if (focusIn(containerElement, 1 /* First */) === 0 /* Error */) {
console.warn("There are no focusable elements inside the <FocusTrap />");
}
}
previousActiveElement.current = ownerDocument == null ? void 0 : ownerDocument.activeElement;
});
}, [enabled]);
return previousActiveElement;
}
function useFocusLock({
ownerDocument,
container,
containers,
previousActiveElement
}, enabled) {
let mounted = useIsMounted();
useEventListener(
ownerDocument == null ? void 0 : ownerDocument.defaultView,
"focus",
(event) => {
if (!enabled)
return;
if (!mounted.current)
return;
let allContainers = resolveContainers(containers);
if (container.current instanceof HTMLElement)
allContainers.add(container.current);
let previous = previousActiveElement.current;
if (!previous)
return;
let toElement = event.target;
if (toElement && toElement instanceof HTMLElement) {
if (!contains(allContainers, toElement)) {
event.preventDefault();
event.stopPropagation();
focusElement(previous);
} else {
previousActiveElement.current = toElement;
focusElement(toElement);
}
} else {
focusElement(previousActiveElement.current);
}
},
true
);
}
function contains(containers, element) {
for (let container of containers) {
if (container.contains(element))
return true;
}
return false;
}
// src/components/portal/portal.tsx
var import_react26 = __toESM(require("react"), 1);
var import_react_dom = require("react-dom");
// src/internal/portal-force-root.tsx
var import_react25 = __toESM(require("react"), 1);
var ForcePortalRootContext = (0, import_react25.createContext)(false);
function usePortalRoot() {
return (0, import_react25.useContext)(ForcePortalRootContext);
}
function ForcePortalRoot(props) {
return /* @__PURE__ */ import_react25.default.createElement(ForcePortalRootContext.Provider, { value: props.force }, props.children);
}
// src/components/portal/portal.tsx
function usePortalTarget(ref) {
let forceInRoot = usePortalRoot();
let groupTarget = (0, import_react26.useContext)(PortalGroupContext);
let ownerDocument = useOwnerDocument(ref);
let [target, setTarget] = (0, import_react26.useState)(() => {
if (!forceInRoot && groupTarget !== null)
return null;
if (env.isServer)
return null;
let existingRoot = ownerDocument == null ? void 0 : ownerDocument.getElementById("headlessui-portal-root");
if (existingRoot)
return existingRoot;
if (ownerDocument === null)
return null;
let root = ownerDocument.createElement("div");
root.setAttribute("id", "headlessui-portal-root");
return ownerDocument.body.appendChild(root);
});
(0, import_react26.useEffect)(() => {
if (target === null)
return;
if (!(ownerDocument == null ? void 0 : ownerDocument.body.contains(target))) {
ownerDocument == null ? void 0 : ownerDocument.body.appendChild(target);
}
}, [target, ownerDocument]);
(0, import_react26.useEffect)(() => {
if (forceInRoot)
return;
if (groupTarget === null)
return;
setTarget(groupTarget.current);
}, [groupTarget, setTarget, forceInRoot]);
return target;
}
var DEFAULT_PORTAL_TAG = import_react26.Fragment;
function PortalFn(props, ref) {
let theirProps = props;
let internalPortalRootRef = (0, import_react26.useRef)(null);
let portalRef = useSyncRefs(
optionalRef((ref2) => {
internalPortalRootRef.current = ref2;
}),
ref
);
let ownerDocument = useOwnerDocument(internalPortalRootRef);
let target = usePortalTarget(internalPortalRootRef);
let [element] = (0, import_react26.useState)(
() => {
var _a3;
return env.isServer ? null : (_a3 = ownerDocument == null ? void 0 : ownerDocument.createElement("div")) != null ? _a3 : null;
}
);
let parent = (0, import_react26.useContext)(PortalParentContext);
let ready = useServerHandoffComplete();
useIsoMorphicEffect(() => {
if (!target || !element)
return;
if (!target.contains(element)) {
element.setAttribute("data-headlessui-portal", "");
target.appendChild(element);
}
}, [target, element]);
useIsoMorphicEffect(() => {
if (!element)
return;
if (!parent)
return;
return parent.register(element);
}, [parent, element]);
useOnUnmount(() => {
var _a3;
if (!target || !element)
return;
if (element instanceof Node && target.contains(element)) {
target.removeChild(element);
}
if (target.childNodes.length <= 0) {
(_a3 = target.parentElement) == null ? void 0 : _a3.removeChild(target);
}
});
if (!ready)
return null;
let ourProps = { ref: portalRef };
return !target || !element ? null : (0, import_react_dom.createPortal)(
render({
ourProps,
theirProps,
defaultTag: DEFAULT_PORTAL_TAG,
name: "Portal"
}),
element
);
}
var DEFAULT_GROUP_TAG = import_react26.Fragment;
var PortalGroupContext = (0, import_react26.createContext)(null);
function GroupFn(props, ref) {
let { target, ...theirProps } = props;
let groupRef = useSyncRefs(ref);
let ourProps = { ref: groupRef };
return /* @__PURE__ */ import_react26.default.createElement(PortalGroupContext.Provider, { value: target }, render({
ourProps,
theirProps,
defaultTag: DEFAULT_GROUP_TAG,
name: "Popover.Group"
}));
}
var PortalParentContext = (0, import_react26.createContext)(null);
function useNestedPortals() {
let parent = (0, import_react26.useContext)(PortalParentContext);
let portals = (0, import_react26.useRef)([]);
let register = useEvent((portal) => {
portals.current.push(portal);
if (parent)
parent.register(portal);
return () => unregister(portal);
});
let unregister = useEvent((portal) => {
let idx = portals.current.indexOf(portal);
if (idx !== -1)
portals.current.splice(idx, 1);
if (parent)
parent.unregister(portal);
});
let api = (0, import_react26.useMemo)(
() => ({ register, unregister, portals }),
[register, unregister, portals]
);
return [
portals,
(0, import_react26.useMemo)(() => {
return function PortalWrapper({ children }) {
return /* @__PURE__ */ import_react26.default.createElement(PortalParentContext.Provider, { value: api }, children);
};
}, [api])
];
}
var PortalRoot = forwardRefWithAs(PortalFn);
var Group = forwardRefWithAs(GroupFn);
var Portal = Object.assign(PortalRoot, { Group });
// src/use-sync-external-store-shim/index.ts
var React11 = __toESM(require("react"), 1);
// src/use-sync-external-store-shim/useSyncExternalStoreShimClient.ts
var React10 = __toESM(require("react"), 1);
function isPolyfill(x, y) {
return x === y && (x !== 0 || 1 / x === 1 / y) || x !== x && y !== y;
}
var is = typeof Object.is === "function" ? Object.is : isPolyfill;
var { useState: useState9, useEffect: useEffect15, useLayoutEffect: useLayoutEffect3, useDebugValue } = React10;
var didWarnOld18Alpha = false;
var didWarnUncachedGetSnapshot = false;
function useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot) {
if (true) {
if (!didWarnOld18Alpha) {
if ("startTransition" in React10) {
didWarnOld18Alpha = true;
console.error(
"You are using an outdated, pre-release alpha of React 18 that does not support useSyncExternalStore. The use-sync-external-store shim will not work correctly. Upgrade to a newer pre-release."
);
}
}
}
const value = getSnapshot();
if (true) {
if (!didWarnUncachedGetSnapshot) {
const cachedValue = getSnapshot();
if (!is(value, cachedValue)) {
console.error("The result of getSnapshot should be cached to avoid an infinite loop");
didWarnUncachedGetSnapshot = true;
}
}
}
const [{ inst }, forceUpdate] = useState9({ inst: { value, getSnapshot } });
useLayoutEffect3(() => {
inst.value = value;
inst.getSnapshot = getSnapshot;
if (checkIfSnapshotChanged(inst)) {
forceUpdate({ inst });
}
}, [subscribe, value, getSnapshot]);
useEffect15(() => {
if (checkIfSnapshotChanged(inst)) {
forceUpdate({ inst });
}
const handleStoreChange = () => {
if (checkIfSnapshotChanged(inst)) {
forceUpdate({ inst });
}
};
return subscribe(handleStoreChange);
}, [subscribe]);
useDebugValue(value);
return value;
}
function checkIfSnapshotChanged(inst) {
const latestGetSnapshot = inst.getSnapshot;
const prevValue = inst.value;
try {
const nextValue = latestGetSnapshot();
return !is(prevValue, nextValue);
} catch (error) {
return true;
}
}
// src/use-sync-external-store-shim/useSyncExternalStoreShimServer.ts
function useSyncExternalStore2(subscribe, getSnapshot, getServerSnapshot) {
return getSnapshot();
}
// src/use-sync-external-store-shim/index.ts
var canUseDOM = !!(typeof window !== "undefined" && typeof window.document !== "undefined" && typeof window.document.createElement !== "undefined");
var isServerEnvironment = !canUseDOM;
var shim = isServerEnvironment ? useSyncExternalStore2 : useSyncExternalStore;
var useSyncExternalStore3 = "useSyncExternalStore" in React11 ? ((r) => r.useSyncExternalStore)(React11) : shim;
// src/hooks/use-store.ts
function useStore(store) {
return useSyncExternalStore3(store.subscribe, store.getSnapshot, store.getSnapshot);
}
// src/utils/store.ts
function createStore(initial, actions) {
let state = initial();
let listeners = /* @__PURE__ */ new Set();
return {
getSnapshot() {
return state;
},
subscribe(onChange) {
listeners.add(onChange);
return () => listeners.delete(onChange);
},
dispatch(key, ...args) {
let newState = actions[key].call(state, ...args);
if (newState) {
state = newState;
listeners.forEach((listener) => listener());
}
}
};
}
// src/hooks/document-overflow/adjust-scrollbar-padding.ts
function adjustScrollbarPadding() {
let scrollbarWidthBefore;
return {
before({ doc }) {
var _a3;
let documentElement = doc.documentElement;
let ownerWindow = (_a3 = doc.defaultView) != null ? _a3 : window;
scrollbarWidthBefore = ownerWindow.innerWidth - documentElement.clientWidth;
},
after({ doc, d }) {
let documentElement = doc.documentElement;
let scrollbarWidthAfter = documentElement.clientWidth - documentElement.offsetWidth;
let scrollbarWidth = scrollbarWidthBefore - scrollbarWidthAfter;
d.style(documentElement, "paddingRight", `${scrollbarWidth}px`);
}
};
}
// src/hooks/document-overflow/handle-ios-locking.ts
function handleIOSLocking() {
if (!isIOS()) {
return {};
}
return {
before({ doc, d, meta }) {
function inAllowedContainer(el) {
return meta.containers.flatMap((resolve) => resolve()).some((container) => container.contains(el));
}
d.microTask(() => {
var _a3;
if (window.getComputedStyle(doc.documentElement).scrollBehavior !== "auto") {
let _d = disposables();
_d.style(doc.documentElement, "scrollBehavior", "auto");
d.add(() => d.microTask(() => _d.dispose()));
}
let scrollPosition = (_a3 = window.scrollY) != null ? _a3 : window.pageYOffset;
let scrollToElement = null;
d.addEventListener(
doc,
"click",
(e) => {
if (!(e.target instanceof HTMLElement)) {
return;
}
try {
let anchor = e.target.closest("a");
if (!anchor)
return;
let { hash } = new URL(anchor.href);
let el = doc.querySelector(hash);
if (el && !inAllowedContainer(el)) {
scrollToElement = el;
}
} catch (err) {
}
},
true
);
d.addEventListener(doc, "touchstart", (e) => {
if (e.target instanceof HTMLElement) {
if (inAllowedContainer(e.target)) {
let rootContainer = e.target;
while (rootContainer.parentElement && inAllowedContainer(rootContainer.parentElement)) {
rootContainer = rootContainer.parentElement;
}
d.style(rootContainer, "overscrollBehavior", "contain");
} else {
d.style(e.target, "touchAction", "none");
}
}
});
d.addEventListener(
doc,
"touchmove",
(e) => {
if (e.target instanceof HTMLElement) {
if (inAllowedContainer(e.target)) {
let scrollableParent = e.target;
while (scrollableParent.parentElement && // Assumption: We are always used in a Headless UI Portal. Once we reach the
// portal itself, we can stop crawling up the tree.
scrollableParent.dataset.headlessuiPortal !== "") {
if (scrollableParent.scrollHeight > scrollableParent.clientHeight || scrollableParent.scrollWidth > scrollableParent.clientWidth) {
break;
}
scrollableParent = scrollableParent.parentElement;
}
if (scrollableParent.dataset.headlessuiPortal === "") {
e.preventDefault();
}
} else {
e.preventDefault();
}
}
},
{ passive: false }
);
d.add(() => {
var _a4;
let newScrollPosition = (_a4 = window.scrollY) != null ? _a4 : window.pageYOffset;
if (scrollPosition !== newScrollPosition) {
window.scrollTo(0, scrollPosition);
}
if (scrollToElement && scrollToElement.isConnected) {
scrollToElement.scrollIntoView({ block: "nearest" });
scrollToElement = null;
}
});
});
}
};
}
// src/hooks/document-overflow/prevent-scroll.ts
function preventScroll() {
return {
before({ doc, d }) {
d.style(doc.documentElement, "overflow", "hidden");
}
};
}
// src/hooks/document-overflow/overflow-store.ts
function buildMeta(fns) {
let tmp = {};
for (let fn of fns) {
Object.assign(tmp, fn(tmp));
}
return tmp;
}
var overflows = createStore(() => /* @__PURE__ */ new Map(), {
PUSH(doc, meta) {
var _a3;
let entry = (_a3 = this.get(doc)) != null ? _a3 : {
doc,
count: 0,
d: disposables(),
meta: /* @__PURE__ */ new Set()
};
entry.count++;
entry.meta.add(meta);
this.set(doc, entry);
return this;
},
POP(doc, meta) {
let entry = this.get(doc);
if (entry) {
entry.count--;
entry.meta.delete(meta);
}
return this;
},
SCROLL_PREVENT({ doc, d, meta }) {
let ctx = {
doc,
d,
meta: buildMeta(meta)
};
let steps = [
handleIOSLocking(),
adjustScrollbarPadding(),
preventScroll()
];
steps.forEach(({ before }) => before == null ? void 0 : before(ctx));
steps.forEach(({ after }) => after == null ? void 0 : after(ctx));
},
SCROLL_ALLOW({ d }) {
d.dispose();
},
TEARDOWN({ doc }) {
this.delete(doc);
}
});
overflows.subscribe(() => {
let docs = overflows.getSnapshot();
let styles = /* @__PURE__ */ new Map();
for (let [doc] of docs) {
styles.set(doc, doc.documentElement.style.overflow);
}
for (let entry of docs.values()) {
let isHidden = styles.get(entry.doc) === "hidden";
let isLocked = entry.count !== 0;
let willChange = isLocked && !isHidden || !isLocked && isHidden;
if (willChange) {
overflows.dispatch(entry.count > 0 ? "SCROLL_PREVENT" : "SCROLL_ALLOW", entry);
}
if (entry.count === 0) {
overflows.dispatch("TEARDOWN", entry);
}
}
});
// src/hooks/document-overflow/use-document-overflow.ts
function useDocumentOverflowLockedEffect(doc, shouldBeLocked, meta) {
let store = useStore(overflows);
let entry = doc ? store.get(doc) : void 0;
let locked = entry ? entry.count > 0 : false;
useIsoMorphicEffect(() => {
if (!doc || !shouldBeLocked) {
return;
}
overflows.dispatch("PUSH", doc, meta);
return () => overflows.dispatch("POP", doc, meta);
}, [shouldBeLocked, doc]);
return locked;
}
// src/hooks/use-inert.tsx
var originals = /* @__PURE__ */ new Map();
var counts = /* @__PURE__ */ new Map();
function useInert(node, enabled = true) {
useIsoMorphicEffect(() => {
var _a3;
if (!enabled)
return;
let element = typeof node === "function" ? node() : node.current;
if (!element)
return;
function cleanup() {
var _a4;
if (!element)
return;
let count2 = (_a4 = counts.get(element)) != null ? _a4 : 1;
if (count2 === 1)
counts.delete(element);
else
counts.set(element, count2 - 1);
if (count2 !== 1)
return;
let original = originals.get(element);
if (!original)
return;
if (original["aria-hidden"] === null)
element.removeAttribute("aria-hidden");
else
element.setAttribute("aria-hidden", original["aria-hidden"]);
element.inert = original.inert;
originals.delete(element);
}
let count = (_a3 = counts.get(element)) != null ? _a3 : 0;
counts.set(element, count + 1);
if (count !== 0)
return cleanup;
originals.set(element, {
"aria-hidden": element.getAttribute("aria-hidden"),
inert: element.inert
});
element.setAttribute("aria-hidden", "true");
element.inert = true;
return cleanup;
}, [node, enabled]);
}
// src/hooks/use-root-containers.tsx
var import_react27 = __toESM(require("react"), 1);
function useRootContainers({
defaultContainers = [],
portals,
mainTreeNodeRef: _mainTreeNodeRef
} = {}) {
var _a3;
let mainTreeNodeRef = (0, import_react27.useRef)((_a3 = _mainTreeNodeRef == null ? void 0 : _mainTreeNodeRef.current) != null ? _a3 : null);
let ownerDocument = useOwnerDocument(mainTreeNodeRef);
let resolveContainers2 = useEvent(() => {
var _a4, _b, _c;
let containers = [];
for (let container of defaultContainers) {
if (container === null)
continue;
if (container instanceof HTMLElement) {
containers.push(container);
} else if ("current" in container && container.current instanceof HTMLElement) {
containers.push(container.current);
}
}
if (portals == null ? void 0 : portals.current) {
for (let portal of portals.current) {
containers.push(portal);
}
}
for (let container of (_a4 = ownerDocument == null ? void 0 : ownerDocument.querySelectorAll("html > *, body > *")) != null ? _a4 : []) {
if (container === document.body)
continue;
if (container === document.head)
continue;
if (!(container instanceof HTMLElement))
continue;
if (container.id === "headlessui-portal-root")
continue;
if (container.contains(mainTreeNodeRef.current))
continue;
if (container.contains((_c = (_b = mainTreeNodeRef.current) == null ? void 0 : _b.getRootNode()) == null ? void 0 : _c.host))
continue;
if (containers.some((defaultContainer) => container.contains(defaultContainer)))
continue;
containers.push(container);
}
return containers;
});
return {
resolveContainers: resolveContainers2,
contains: useEvent(
(element) => resolveContainers2().some((container) => container.contains(element))
),
mainTreeNodeRef,
MainTreeNode: (0, import_react27.useMemo)(() => {
return function MainTreeNode() {
if (_mainTreeNodeRef != null)
return null;
return /* @__PURE__ */ import_react27.default.createElement(Hidden, { features: 4 /* Hidden */, ref: mainTreeNodeRef });
};
}, [mainTreeNodeRef, _mainTreeNodeRef])
};
}
function useMainTreeNode() {
let mainTreeNodeRef = (0, import_react27.useRef)(null);
return {
mainTreeNodeRef,
MainTreeNode: (0, import_react27.useMemo)(() => {
return function MainTreeNode() {
return /* @__PURE__ */ import_react27.default.createElement(Hidden, { features: 4 /* Hidden */, ref: mainTreeNodeRef });
};
}, [mainTreeNodeRef])
};
}
// src/internal/stack-context.tsx
var import_react28 = __toESM(require("react"), 1);
var StackContext = (0, import_react28.createContext)(() => {
});
StackContext.displayName = "StackContext";
function useStackContext() {
return (0, import_react28.useContext)(StackContext);
}
function StackProvider({
children,
onUpdate,
type,
element,
enabled
}) {
let parentUpdate = useStackContext();
let notify = useEvent((...args) => {
onUpdate == null ? void 0 : onUpdate(...args);
parentUpdate(...args);
});
useIsoMorphicEffect(() => {
let shouldNotify = enabled === void 0 || enabled === true;
shouldNotify && notify(0 /* Add */, type, element);
return () => {
shouldNotify && notify(1 /* Remove */, type, element);
};
}, [notify, type, element, enabled]);
return /* @__PURE__ */ import_react28.default.createElement(StackContext.Provider, { value: notify }, children);
}
// src/components/description/description.tsx
var import_react29 = __toESM(require("react"), 1);
var DescriptionContext = (0, import_react29.createContext)(null);
function useDescriptionContext() {
let context = (0, import_react29.useContext)(DescriptionContext);
if (context === null) {
let err = new Error(
"You used a <Description /> component, but it is not inside a relevant parent."
);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useDescriptionContext);
throw err;
}
return context;
}
function useDescriptions() {
let [descriptionIds, setDescriptionIds] = (0, import_react29.useState)([]);
return [
// The actual id's as string or undefined
descriptionIds.length > 0 ? descriptionIds.join(" ") : void 0,
// The provider component
(0, import_react29.useMemo)(() => {
return function DescriptionProvider(props) {
let register = useEvent((value) => {
setDescriptionIds((existing) => [...existing, value]);
return () => setDescriptionIds((existing) => {
let clone = existing.slice();
let idx = clone.indexOf(value);
if (idx !== -1)
clone.splice(idx, 1);
return clone;
});
});
let contextBag = (0, import_react29.useMemo)(
() => ({ register, slot: props.slot, name: props.name, props: props.props }),
[register, props.slot, props.name, props.props]
);
return /* @__PURE__ */ import_react29.default.createElement(DescriptionContext.Provider, { value: contextBag }, props.children);
};
}, [setDescriptionIds])
];
}
var DEFAULT_DESCRIPTION_TAG = "p";
function DescriptionFn(props, ref) {
let internalId = useId();
let { id = `headlessui-description-${internalId}`, ...theirProps } = props;
let context = useDescriptionContext();
let descriptionRef = useSyncRefs(ref);
useIsoMorphicEffect(() => context.register(id), [id, context.register]);
let ourProps = { ref: descriptionRef, ...context.props, id };
return render({
ourProps,
theirProps,
slot: context.slot || {},
defaultTag: DEFAULT_DESCRIPTION_TAG,
name: context.name || "Description"
});
}
var DescriptionRoot = forwardRefWithAs(DescriptionFn);
var Description = Object.assign(DescriptionRoot, {
//
});
// src/components/dialog/dialog.tsx
var reducers2 = {
[0 /* SetTitleId */](state, action) {
if (state.titleId === action.id)
return state;
return { ...state, titleId: action.id };
}
};
var DialogContext = (0, import_react30.createContext)(null);
DialogContext.displayName = "DialogContext";
function useDialogContext(component) {
let context = (0, import_react30.useContext)(DialogContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Dialog /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useDialogContext);
throw err;
}
return context;
}
function useScrollLock(ownerDocument, enabled, resolveAllowedContainers = () => [document.body]) {
useDocumentOverflowLockedEffect(ownerDocument, enabled, (meta) => {
var _a3;
return {
containers: [...(_a3 = meta.containers) != null ? _a3 : [], resolveAllowedContainers]
};
});
}
function stateReducer2(state, action) {
return match(action.type, reducers2, state, action);
}
var DEFAULT_DIALOG_TAG = "div";
var DialogRenderFeatures = 1 /* RenderStrategy */ | 2 /* Static */;
function DialogFn(props, ref) {
let internalId = useId();
let {
id = `headlessui-dialog-${internalId}`,
open,
onClose,
initialFocus,
role = "dialog",
__demoMode = false,
...theirProps
} = props;
let [nestedDialogCount, setNestedDialogCount] = (0, import_react30.useState)(0);
let didWarnOnRole = (0, import_react30.useRef)(false);
role = function() {
if (role === "dialog" || role === "alertdialog") {
return role;
}
if (!didWarnOnRole.current) {
didWarnOnRole.current = true;
console.warn(
`Invalid role [${role}] passed to <Dialog />. Only \`dialog\` and and \`alertdialog\` are supported. Using \`dialog\` instead.`
);
}
return "dialog";
}();
let usesOpenClosedState = useOpenClosed();
if (open === void 0 && usesOpenClosedState !== null) {
open = (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
let internalDialogRef = (0, import_react30.useRef)(null);
let dialogRef = useSyncRefs(internalDialogRef, ref);
let ownerDocument = useOwnerDocument(internalDialogRef);
let hasOpen = props.hasOwnProperty("open") || usesOpenClosedState !== null;
let hasOnClose = props.hasOwnProperty("onClose");
if (!hasOpen && !hasOnClose) {
throw new Error(
`You have to provide an \`open\` and an \`onClose\` prop to the \`Dialog\` component.`
);
}
if (!hasOpen) {
throw new Error(
`You provided an \`onClose\` prop to the \`Dialog\`, but forgot an \`open\` prop.`
);
}
if (!hasOnClose) {
throw new Error(
`You provided an \`open\` prop to the \`Dialog\`, but forgot an \`onClose\` prop.`
);
}
if (typeof open !== "boolean") {
throw new Error(
`You provided an \`open\` prop to the \`Dialog\`, but the value is not a boolean. Received: ${open}`
);
}
if (typeof onClose !== "function") {
throw new Error(
`You provided an \`onClose\` prop to the \`Dialog\`, but the value is not a function. Received: ${onClose}`
);
}
let dialogState = open ? 0 /* Open */ : 1 /* Closed */;
let [state, dispatch] = (0, import_react30.useReducer)(stateReducer2, {
titleId: null,
descriptionId: null,
panelRef: (0, import_react30.createRef)()
});
let close = useEvent(() => onClose(false));
let setTitleId = useEvent((id2) => dispatch({ type: 0 /* SetTitleId */, id: id2 }));
let ready = useServerHandoffComplete();
let enabled = ready ? __demoMode ? false : dialogState === 0 /* Open */ : false;
let hasNestedDialogs = nestedDialogCount > 1;
let hasParentDialog = (0, import_react30.useContext)(DialogContext) !== null;
let [portals, PortalWrapper] = useNestedPortals();
let defaultContainer = {
get current() {
var _a3;
return (_a3 = state.panelRef.current) != null ? _a3 : internalDialogRef.current;
}
};
let {
resolveContainers: resolveRootContainers,
mainTreeNodeRef,
MainTreeNode
} = useRootContainers({
portals,
defaultContainers: [defaultContainer]
});
let position = !hasNestedDialogs ? "leaf" : "parent";
let isClosing = usesOpenClosedState !== null ? (usesOpenClosedState & 4 /* Closing */) === 4 /* Closing */ : false;
let inertOthersEnabled = (() => {
if (hasParentDialog)
return false;
if (isClosing)
return false;
return enabled;
})();
let resolveRootOfMainTreeNode = (0, import_react30.useCallback)(() => {
var _a3, _b;
return (_b = Array.from((_a3 = ownerDocument == null ? void 0 : ownerDocument.querySelectorAll("body > *")) != null ? _a3 : []).find((root) => {
if (root.id === "headlessui-portal-root")
return false;
return root.contains(mainTreeNodeRef.current) && root instanceof HTMLElement;
})) != null ? _b : null;
}, [mainTreeNodeRef]);
useInert(resolveRootOfMainTreeNode, inertOthersEnabled);
let inertParentDialogs = (() => {
if (hasNestedDialogs)
return true;
return enabled;
})();
let resolveRootOfParentDialog = (0, import_react30.useCallback)(() => {
var _a3, _b;
return (_b = Array.from((_a3 = ownerDocument == null ? void 0 : ownerDocument.querySelectorAll("[data-headlessui-portal]")) != null ? _a3 : []).find(
(root) => root.contains(mainTreeNodeRef.current) && root instanceof HTMLElement
)) != null ? _b : null;
}, [mainTreeNodeRef]);
useInert(resolveRootOfParentDialog, inertParentDialogs);
let outsideClickEnabled = (() => {
if (!enabled)
return false;
if (hasNestedDialogs)
return false;
return true;
})();
useOutsideClick(
resolveRootContainers,
(event) => {
event.preventDefault();
close();
},
outsideClickEnabled
);
let escapeToCloseEnabled = (() => {
if (hasNestedDialogs)
return false;
if (dialogState !== 0 /* Open */)
return false;
return true;
})();
useEventListener(ownerDocument == null ? void 0 : ownerDocument.defaultView, "keydown", (event) => {
if (!escapeToCloseEnabled)
return;
if (event.defaultPrevented)
return;
if (event.key !== "Escape" /* Escape */)
return;
event.preventDefault();
event.stopPropagation();
close();
});
let scrollLockEnabled = (() => {
if (isClosing)
return false;
if (dialogState !== 0 /* Open */)
return false;
if (hasParentDialog)
return false;
return true;
})();
useScrollLock(ownerDocument, scrollLockEnabled, resolveRootContainers);
(0, import_react30.useEffect)(() => {
if (dialogState !== 0 /* Open */)
return;
if (!internalDialogRef.current)
return;
let observer = new ResizeObserver((entries) => {
for (let entry of entries) {
let rect = entry.target.getBoundingClientRect();
if (rect.x === 0 && rect.y === 0 && rect.width === 0 && rect.height === 0) {
close();
}
}
});
observer.observe(internalDialogRef.current);
return () => observer.disconnect();
}, [dialogState, internalDialogRef, close]);
let [describedby, DescriptionProvider] = useDescriptions();
let contextBag = (0, import_react30.useMemo)(
() => [{ dialogState, close, setTitleId }, state],
[dialogState, state, close, setTitleId]
);
let slot = (0, import_react30.useMemo)(
() => ({ open: dialogState === 0 /* Open */ }),
[dialogState]
);
let ourProps = {
ref: dialogRef,
id,
role,
"aria-modal": dialogState === 0 /* Open */ ? true : void 0,
"aria-labelledby": state.titleId,
"aria-describedby": describedby
};
return /* @__PURE__ */ import_react30.default.createElement(
StackProvider,
{
type: "Dialog",
enabled: dialogState === 0 /* Open */,
element: internalDialogRef,
onUpdate: useEvent((message, type) => {
if (type !== "Dialog")
return;
match(message, {
[0 /* Add */]: () => setNestedDialogCount((count) => count + 1),
[1 /* Remove */]: () => setNestedDialogCount((count) => count - 1)
});
})
},
/* @__PURE__ */ import_react30.default.createElement(ForcePortalRoot, { force: true }, /* @__PURE__ */ import_react30.default.createElement(Portal, null, /* @__PURE__ */ import_react30.default.createElement(DialogContext.Provider, { value: contextBag }, /* @__PURE__ */ import_react30.default.createElement(Portal.Group, { target: internalDialogRef }, /* @__PURE__ */ import_react30.default.createElement(ForcePortalRoot, { force: false }, /* @__PURE__ */ import_react30.default.createElement(DescriptionProvider, { slot, name: "Dialog.Description" }, /* @__PURE__ */ import_react30.default.createElement(
FocusTrap,
{
initialFocus,
containers: resolveRootContainers,
features: enabled ? match(position, {
parent: FocusTrap.features.RestoreFocus,
leaf: FocusTrap.features.All & ~FocusTrap.features.FocusLock
}) : FocusTrap.features.None
},
/* @__PURE__ */ import_react30.default.createElement(PortalWrapper, null, render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_DIALOG_TAG,
features: DialogRenderFeatures,
visible: dialogState === 0 /* Open */,
name: "Dialog"
}))
))))))),
/* @__PURE__ */ import_react30.default.createElement(MainTreeNode, null)
);
}
var DEFAULT_OVERLAY_TAG = "div";
function OverlayFn(props, ref) {
let internalId = useId();
let { id = `headlessui-dialog-overlay-${internalId}`, ...theirProps } = props;
let [{ dialogState, close }] = useDialogContext("Dialog.Overlay");
let overlayRef = useSyncRefs(ref);
let handleClick = useEvent((event) => {
if (event.target !== event.currentTarget)
return;
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
event.preventDefault();
event.stopPropagation();
close();
});
let slot = (0, import_react30.useMemo)(
() => ({ open: dialogState === 0 /* Open */ }),
[dialogState]
);
let ourProps = {
ref: overlayRef,
id,
"aria-hidden": true,
onClick: handleClick
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OVERLAY_TAG,
name: "Dialog.Overlay"
});
}
var DEFAULT_BACKDROP_TAG = "div";
function BackdropFn(props, ref) {
let internalId = useId();
let { id = `headlessui-dialog-backdrop-${internalId}`, ...theirProps } = props;
let [{ dialogState }, state] = useDialogContext("Dialog.Backdrop");
let backdropRef = useSyncRefs(ref);
(0, import_react30.useEffect)(() => {
if (state.panelRef.current === null) {
throw new Error(
`A <Dialog.Backdrop /> component is being used, but a <Dialog.Panel /> component is missing.`
);
}
}, [state.panelRef]);
let slot = (0, import_react30.useMemo)(
() => ({ open: dialogState === 0 /* Open */ }),
[dialogState]
);
let ourProps = {
ref: backdropRef,
id,
"aria-hidden": true
};
return /* @__PURE__ */ import_react30.default.createElement(ForcePortalRoot, { force: true }, /* @__PURE__ */ import_react30.default.createElement(Portal, null, render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_BACKDROP_TAG,
name: "Dialog.Backdrop"
})));
}
var DEFAULT_PANEL_TAG = "div";
function PanelFn(props, ref) {
let internalId = useId();
let { id = `headlessui-dialog-panel-${internalId}`, ...theirProps } = props;
let [{ dialogState }, state] = useDialogContext("Dialog.Panel");
let panelRef = useSyncRefs(ref, state.panelRef);
let slot = (0, import_react30.useMemo)(
() => ({ open: dialogState === 0 /* Open */ }),
[dialogState]
);
let handleClick = useEvent((event) => {
event.stopPropagation();
});
let ourProps = {
ref: panelRef,
id,
onClick: handleClick
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_PANEL_TAG,
name: "Dialog.Panel"
});
}
var DEFAULT_TITLE_TAG = "h2";
function TitleFn(props, ref) {
let internalId = useId();
let { id = `headlessui-dialog-title-${internalId}`, ...theirProps } = props;
let [{ dialogState, setTitleId }] = useDialogContext("Dialog.Title");
let titleRef = useSyncRefs(ref);
(0, import_react30.useEffect)(() => {
setTitleId(id);
return () => setTitleId(null);
}, [id, setTitleId]);
let slot = (0, import_react30.useMemo)(
() => ({ open: dialogState === 0 /* Open */ }),
[dialogState]
);
let ourProps = { ref: titleRef, id };
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_TITLE_TAG,
name: "Dialog.Title"
});
}
var DialogRoot = forwardRefWithAs(DialogFn);
var Backdrop = forwardRefWithAs(BackdropFn);
var Panel = forwardRefWithAs(PanelFn);
var Overlay = forwardRefWithAs(OverlayFn);
var Title = forwardRefWithAs(TitleFn);
var Dialog = Object.assign(DialogRoot, {
Backdrop,
Panel,
Overlay,
Title,
Description
});
// src/components/disclosure/disclosure.tsx
var import_react32 = __toESM(require("react"), 1);
// src/utils/start-transition.ts
var import_react31 = __toESM(require("react"), 1);
var _a2;
var startTransition = (
// Prefer React's `startTransition` if it's available.
// @ts-expect-error - `startTransition` doesn't exist in React < 18.
(_a2 = import_react31.default.startTransition) != null ? _a2 : function startTransition2(cb) {
cb();
}
);
// src/components/disclosure/disclosure.tsx
var reducers3 = {
[0 /* ToggleDisclosure */]: (state) => ({
...state,
disclosureState: match(state.disclosureState, {
[0 /* Open */]: 1 /* Closed */,
[1 /* Closed */]: 0 /* Open */
})
}),
[1 /* CloseDisclosure */]: (state) => {
if (state.disclosureState === 1 /* Closed */)
return state;
return { ...state, disclosureState: 1 /* Closed */ };
},
[4 /* LinkPanel */](state) {
if (state.linkedPanel === true)
return state;
return { ...state, linkedPanel: true };
},
[5 /* UnlinkPanel */](state) {
if (state.linkedPanel === false)
return state;
return { ...state, linkedPanel: false };
},
[2 /* SetButtonId */](state, action) {
if (state.buttonId === action.buttonId)
return state;
return { ...state, buttonId: action.buttonId };
},
[3 /* SetPanelId */](state, action) {
if (state.panelId === action.panelId)
return state;
return { ...state, panelId: action.panelId };
}
};
var DisclosureContext = (0, import_react32.createContext)(null);
DisclosureContext.displayName = "DisclosureContext";
function useDisclosureContext(component) {
let context = (0, import_react32.useContext)(DisclosureContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Disclosure /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useDisclosureContext);
throw err;
}
return context;
}
var DisclosureAPIContext = (0, import_react32.createContext)(null);
DisclosureAPIContext.displayName = "DisclosureAPIContext";
function useDisclosureAPIContext(component) {
let context = (0, import_react32.useContext)(DisclosureAPIContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Disclosure /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useDisclosureAPIContext);
throw err;
}
return context;
}
var DisclosurePanelContext = (0, import_react32.createContext)(null);
DisclosurePanelContext.displayName = "DisclosurePanelContext";
function useDisclosurePanelContext() {
return (0, import_react32.useContext)(DisclosurePanelContext);
}
function stateReducer3(state, action) {
return match(action.type, reducers3, state, action);
}
var DEFAULT_DISCLOSURE_TAG = import_react32.Fragment;
function DisclosureFn(props, ref) {
let { defaultOpen = false, ...theirProps } = props;
let internalDisclosureRef = (0, import_react32.useRef)(null);
let disclosureRef = useSyncRefs(
ref,
optionalRef(
(ref2) => {
internalDisclosureRef.current = ref2;
},
props.as === void 0 || // @ts-expect-error The `as` prop _can_ be a Fragment
props.as === import_react32.Fragment
)
);
let panelRef = (0, import_react32.useRef)(null);
let buttonRef = (0, import_react32.useRef)(null);
let reducerBag = (0, import_react32.useReducer)(stateReducer3, {
disclosureState: defaultOpen ? 0 /* Open */ : 1 /* Closed */,
linkedPanel: false,
buttonRef,
panelRef,
buttonId: null,
panelId: null
});
let [{ disclosureState, buttonId }, dispatch] = reducerBag;
let close = useEvent((focusableElement) => {
dispatch({ type: 1 /* CloseDisclosure */ });
let ownerDocument = getOwnerDocument(internalDisclosureRef);
if (!ownerDocument)
return;
if (!buttonId)
return;
let restoreElement = (() => {
if (!focusableElement)
return ownerDocument.getElementById(buttonId);
if (focusableElement instanceof HTMLElement)
return focusableElement;
if (focusableElement.current instanceof HTMLElement)
return focusableElement.current;
return ownerDocument.getElementById(buttonId);
})();
restoreElement == null ? void 0 : restoreElement.focus();
});
let api = (0, import_react32.useMemo)(() => ({ close }), [close]);
let slot = (0, import_react32.useMemo)(
() => ({ open: disclosureState === 0 /* Open */, close }),
[disclosureState, close]
);
let ourProps = {
ref: disclosureRef
};
return /* @__PURE__ */ import_react32.default.createElement(DisclosureContext.Provider, { value: reducerBag }, /* @__PURE__ */ import_react32.default.createElement(DisclosureAPIContext.Provider, { value: api }, /* @__PURE__ */ import_react32.default.createElement(
OpenClosedProvider,
{
value: match(disclosureState, {
[0 /* Open */]: 1 /* Open */,
[1 /* Closed */]: 2 /* Closed */
})
},
render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_DISCLOSURE_TAG,
name: "Disclosure"
})
)));
}
var DEFAULT_BUTTON_TAG2 = "button";
function ButtonFn2(props, ref) {
let internalId = useId();
let { id = `headlessui-disclosure-button-${internalId}`, ...theirProps } = props;
let [state, dispatch] = useDisclosureContext("Disclosure.Button");
let panelContext = useDisclosurePanelContext();
let isWithinPanel = panelContext === null ? false : panelContext === state.panelId;
let internalButtonRef = (0, import_react32.useRef)(null);
let buttonRef = useSyncRefs(internalButtonRef, ref, !isWithinPanel ? state.buttonRef : null);
let mergeRefs = useMergeRefsFn();
(0, import_react32.useEffect)(() => {
if (isWithinPanel)
return;
dispatch({ type: 2 /* SetButtonId */, buttonId: id });
return () => {
dispatch({ type: 2 /* SetButtonId */, buttonId: null });
};
}, [id, dispatch, isWithinPanel]);
let handleKeyDown = useEvent((event) => {
var _a3;
if (isWithinPanel) {
if (state.disclosureState === 1 /* Closed */)
return;
switch (event.key) {
case " " /* Space */:
case "Enter" /* Enter */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 0 /* ToggleDisclosure */ });
(_a3 = state.buttonRef.current) == null ? void 0 : _a3.focus();
break;
}
} else {
switch (event.key) {
case " " /* Space */:
case "Enter" /* Enter */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 0 /* ToggleDisclosure */ });
break;
}
}
});
let handleKeyUp = useEvent((event) => {
switch (event.key) {
case " " /* Space */:
event.preventDefault();
break;
}
});
let handleClick = useEvent((event) => {
var _a3;
if (isDisabledReactIssue7711(event.currentTarget))
return;
if (props.disabled)
return;
if (isWithinPanel) {
dispatch({ type: 0 /* ToggleDisclosure */ });
(_a3 = state.buttonRef.current) == null ? void 0 : _a3.focus();
} else {
dispatch({ type: 0 /* ToggleDisclosure */ });
}
});
let slot = (0, import_react32.useMemo)(
() => ({ open: state.disclosureState === 0 /* Open */ }),
[state]
);
let type = useResolveButtonType(props, internalButtonRef);
let ourProps = isWithinPanel ? { ref: buttonRef, type, onKeyDown: handleKeyDown, onClick: handleClick } : {
ref: buttonRef,
id,
type,
"aria-expanded": state.disclosureState === 0 /* Open */,
"aria-controls": state.linkedPanel ? state.panelId : void 0,
onKeyDown: handleKeyDown,
onKeyUp: handleKeyUp,
onClick: handleClick
};
return render({
mergeRefs,
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_BUTTON_TAG2,
name: "Disclosure.Button"
});
}
var DEFAULT_PANEL_TAG2 = "div";
var PanelRenderFeatures = 1 /* RenderStrategy */ | 2 /* Static */;
function PanelFn2(props, ref) {
let internalId = useId();
let { id = `headlessui-disclosure-panel-${internalId}`, ...theirProps } = props;
let [state, dispatch] = useDisclosureContext("Disclosure.Panel");
let { close } = useDisclosureAPIContext("Disclosure.Panel");
let mergeRefs = useMergeRefsFn();
let panelRef = useSyncRefs(ref, state.panelRef, (el) => {
startTransition(() => dispatch({ type: el ? 4 /* LinkPanel */ : 5 /* UnlinkPanel */ }));
});
(0, import_react32.useEffect)(() => {
dispatch({ type: 3 /* SetPanelId */, panelId: id });
return () => {
dispatch({ type: 3 /* SetPanelId */, panelId: null });
};
}, [id, dispatch]);
let usesOpenClosedState = useOpenClosed();
let visible = (() => {
if (usesOpenClosedState !== null) {
return (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
return state.disclosureState === 0 /* Open */;
})();
let slot = (0, import_react32.useMemo)(
() => ({ open: state.disclosureState === 0 /* Open */, close }),
[state, close]
);
let ourProps = {
ref: panelRef,
id
};
return /* @__PURE__ */ import_react32.default.createElement(DisclosurePanelContext.Provider, { value: state.panelId }, render({
mergeRefs,
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_PANEL_TAG2,
features: PanelRenderFeatures,
visible,
name: "Disclosure.Panel"
}));
}
var DisclosureRoot = forwardRefWithAs(DisclosureFn);
var Button2 = forwardRefWithAs(ButtonFn2);
var Panel2 = forwardRefWithAs(PanelFn2);
var Disclosure = Object.assign(DisclosureRoot, { Button: Button2, Panel: Panel2 });
// src/components/listbox/listbox.tsx
var import_react34 = __toESM(require("react"), 1);
// src/hooks/use-text-value.ts
var import_react33 = require("react");
// src/utils/get-text-value.ts
var emojiRegex = /([\u2700-\u27BF]|[\uE000-\uF8FF]|\uD83C[\uDC00-\uDFFF]|\uD83D[\uDC00-\uDFFF]|[\u2011-\u26FF]|\uD83E[\uDD10-\uDDFF])/g;
function getTextContents(element) {
var _a3, _b;
let currentInnerText = (_a3 = element.innerText) != null ? _a3 : "";
let copy = element.cloneNode(true);
if (!(copy instanceof HTMLElement)) {
return currentInnerText;
}
let dropped = false;
for (let child of copy.querySelectorAll('[hidden],[aria-hidden],[role="img"]')) {
child.remove();
dropped = true;
}
let value = dropped ? (_b = copy.innerText) != null ? _b : "" : currentInnerText;
if (emojiRegex.test(value)) {
value = value.replace(emojiRegex, "");
}
return value;
}
function getTextValue(element) {
let label = element.getAttribute("aria-label");
if (typeof label === "string")
return label.trim();
let labelledby = element.getAttribute("aria-labelledby");
if (labelledby) {
let labels = labelledby.split(" ").map((labelledby2) => {
let labelEl = document.getElementById(labelledby2);
if (labelEl) {
let label2 = labelEl.getAttribute("aria-label");
if (typeof label2 === "string")
return label2.trim();
return getTextContents(labelEl).trim();
}
return null;
}).filter(Boolean);
if (labels.length > 0)
return labels.join(", ");
}
return getTextContents(element).trim();
}
// src/hooks/use-text-value.ts
function useTextValue(element) {
let cacheKey = (0, import_react33.useRef)("");
let cacheValue = (0, import_react33.useRef)("");
return useEvent(() => {
let el = element.current;
if (!el)
return "";
let currentKey = el.innerText;
if (cacheKey.current === currentKey) {
return cacheValue.current;
}
let value = getTextValue(el).trim().toLowerCase();
cacheKey.current = currentKey;
cacheValue.current = value;
return value;
});
}
// src/components/listbox/listbox.tsx
function adjustOrderedState2(state, adjustment = (i) => i) {
let currentActiveOption = state.activeOptionIndex !== null ? state.options[state.activeOptionIndex] : null;
let sortedOptions = sortByDomNode(
adjustment(state.options.slice()),
(option) => option.dataRef.current.domRef.current
);
let adjustedActiveOptionIndex = currentActiveOption ? sortedOptions.indexOf(currentActiveOption) : null;
if (adjustedActiveOptionIndex === -1) {
adjustedActiveOptionIndex = null;
}
return {
options: sortedOptions,
activeOptionIndex: adjustedActiveOptionIndex
};
}
var reducers4 = {
[1 /* CloseListbox */](state) {
if (state.dataRef.current.disabled)
return state;
if (state.listboxState === 1 /* Closed */)
return state;
return { ...state, activeOptionIndex: null, listboxState: 1 /* Closed */ };
},
[0 /* OpenListbox */](state) {
if (state.dataRef.current.disabled)
return state;
if (state.listboxState === 0 /* Open */)
return state;
let activeOptionIndex = state.activeOptionIndex;
let { isSelected } = state.dataRef.current;
let optionIdx = state.options.findIndex((option) => isSelected(option.dataRef.current.value));
if (optionIdx !== -1) {
activeOptionIndex = optionIdx;
}
return { ...state, listboxState: 0 /* Open */, activeOptionIndex };
},
[2 /* GoToOption */](state, action) {
var _a3;
if (state.dataRef.current.disabled)
return state;
if (state.listboxState === 1 /* Closed */)
return state;
let adjustedState = adjustOrderedState2(state);
let activeOptionIndex = calculateActiveIndex(action, {
resolveItems: () => adjustedState.options,
resolveActiveIndex: () => adjustedState.activeOptionIndex,
resolveId: (option) => option.id,
resolveDisabled: (option) => option.dataRef.current.disabled
});
return {
...state,
...adjustedState,
searchQuery: "",
activeOptionIndex,
activationTrigger: (_a3 = action.trigger) != null ? _a3 : 1 /* Other */
};
},
[3 /* Search */]: (state, action) => {
if (state.dataRef.current.disabled)
return state;
if (state.listboxState === 1 /* Closed */)
return state;
let wasAlreadySearching = state.searchQuery !== "";
let offset = wasAlreadySearching ? 0 : 1;
let searchQuery = state.searchQuery + action.value.toLowerCase();
let reOrderedOptions = state.activeOptionIndex !== null ? state.options.slice(state.activeOptionIndex + offset).concat(state.options.slice(0, state.activeOptionIndex + offset)) : state.options;
let matchingOption = reOrderedOptions.find(
(option) => {
var _a3;
return !option.dataRef.current.disabled && ((_a3 = option.dataRef.current.textValue) == null ? void 0 : _a3.startsWith(searchQuery));
}
);
let matchIdx = matchingOption ? state.options.indexOf(matchingOption) : -1;
if (matchIdx === -1 || matchIdx === state.activeOptionIndex)
return { ...state, searchQuery };
return {
...state,
searchQuery,
activeOptionIndex: matchIdx,
activationTrigger: 1 /* Other */
};
},
[4 /* ClearSearch */](state) {
if (state.dataRef.current.disabled)
return state;
if (state.listboxState === 1 /* Closed */)
return state;
if (state.searchQuery === "")
return state;
return { ...state, searchQuery: "" };
},
[5 /* RegisterOption */]: (state, action) => {
let option = { id: action.id, dataRef: action.dataRef };
let adjustedState = adjustOrderedState2(state, (options) => [...options, option]);
if (state.activeOptionIndex === null) {
if (state.dataRef.current.isSelected(action.dataRef.current.value)) {
adjustedState.activeOptionIndex = adjustedState.options.indexOf(option);
}
}
return { ...state, ...adjustedState };
},
[6 /* UnregisterOption */]: (state, action) => {
let adjustedState = adjustOrderedState2(state, (options) => {
let idx = options.findIndex((a) => a.id === action.id);
if (idx !== -1)
options.splice(idx, 1);
return options;
});
return {
...state,
...adjustedState,
activationTrigger: 1 /* Other */
};
},
[7 /* RegisterLabel */]: (state, action) => {
return {
...state,
labelId: action.id
};
}
};
var ListboxActionsContext = (0, import_react34.createContext)(null);
ListboxActionsContext.displayName = "ListboxActionsContext";
function useActions2(component) {
let context = (0, import_react34.useContext)(ListboxActionsContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Listbox /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useActions2);
throw err;
}
return context;
}
var ListboxDataContext = (0, import_react34.createContext)(null);
ListboxDataContext.displayName = "ListboxDataContext";
function useData2(component) {
let context = (0, import_react34.useContext)(ListboxDataContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Listbox /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useData2);
throw err;
}
return context;
}
function stateReducer4(state, action) {
return match(action.type, reducers4, state, action);
}
var DEFAULT_LISTBOX_TAG = import_react34.Fragment;
function ListboxFn(props, ref) {
let {
value: controlledValue,
defaultValue,
form: formName,
name,
onChange: controlledOnChange,
by = (a, z) => a === z,
disabled = false,
horizontal = false,
multiple = false,
...theirProps
} = props;
const orientation = horizontal ? "horizontal" : "vertical";
let listboxRef = useSyncRefs(ref);
let [value = multiple ? [] : void 0, theirOnChange] = useControllable(
controlledValue,
controlledOnChange,
defaultValue
);
let [state, dispatch] = (0, import_react34.useReducer)(stateReducer4, {
dataRef: (0, import_react34.createRef)(),
listboxState: 1 /* Closed */,
options: [],
searchQuery: "",
labelId: null,
activeOptionIndex: null,
activationTrigger: 1 /* Other */
});
let optionsPropsRef = (0, import_react34.useRef)({ static: false, hold: false });
let labelRef = (0, import_react34.useRef)(null);
let buttonRef = (0, import_react34.useRef)(null);
let optionsRef = (0, import_react34.useRef)(null);
let compare = useEvent(
typeof by === "string" ? (a, z) => {
let property = by;
return (a == null ? void 0 : a[property]) === (z == null ? void 0 : z[property]);
} : by
);
let isSelected = (0, import_react34.useCallback)(
(compareValue) => match(data.mode, {
[1 /* Multi */]: () => value.some((option) => compare(option, compareValue)),
[0 /* Single */]: () => compare(value, compareValue)
}),
[value]
);
let data = (0, import_react34.useMemo)(
() => ({
...state,
value,
disabled,
mode: multiple ? 1 /* Multi */ : 0 /* Single */,
orientation,
compare,
isSelected,
optionsPropsRef,
labelRef,
buttonRef,
optionsRef
}),
[value, disabled, multiple, state]
);
useIsoMorphicEffect(() => {
state.dataRef.current = data;
}, [data]);
useOutsideClick(
[data.buttonRef, data.optionsRef],
(event, target) => {
var _a3;
dispatch({ type: 1 /* CloseListbox */ });
if (!isFocusableElement(target, 1 /* Loose */)) {
event.preventDefault();
(_a3 = data.buttonRef.current) == null ? void 0 : _a3.focus();
}
},
data.listboxState === 0 /* Open */
);
let slot = (0, import_react34.useMemo)(
() => ({ open: data.listboxState === 0 /* Open */, disabled, value }),
[data, disabled, value]
);
let selectOption = useEvent((id) => {
let option = data.options.find((item) => item.id === id);
if (!option)
return;
onChange(option.dataRef.current.value);
});
let selectActiveOption = useEvent(() => {
if (data.activeOptionIndex !== null) {
let { dataRef, id } = data.options[data.activeOptionIndex];
onChange(dataRef.current.value);
dispatch({ type: 2 /* GoToOption */, focus: 4 /* Specific */, id });
}
});
let openListbox = useEvent(() => dispatch({ type: 0 /* OpenListbox */ }));
let closeListbox = useEvent(() => dispatch({ type: 1 /* CloseListbox */ }));
let goToOption = useEvent((focus, id, trigger) => {
if (focus === 4 /* Specific */) {
return dispatch({ type: 2 /* GoToOption */, focus: 4 /* Specific */, id, trigger });
}
return dispatch({ type: 2 /* GoToOption */, focus, trigger });
});
let registerOption = useEvent((id, dataRef) => {
dispatch({ type: 5 /* RegisterOption */, id, dataRef });
return () => dispatch({ type: 6 /* UnregisterOption */, id });
});
let registerLabel = useEvent((id) => {
dispatch({ type: 7 /* RegisterLabel */, id });
return () => dispatch({ type: 7 /* RegisterLabel */, id: null });
});
let onChange = useEvent((value2) => {
return match(data.mode, {
[0 /* Single */]() {
return theirOnChange == null ? void 0 : theirOnChange(value2);
},
[1 /* Multi */]() {
let copy = data.value.slice();
let idx = copy.findIndex((item) => compare(item, value2));
if (idx === -1) {
copy.push(value2);
} else {
copy.splice(idx, 1);
}
return theirOnChange == null ? void 0 : theirOnChange(copy);
}
});
});
let search = useEvent((value2) => dispatch({ type: 3 /* Search */, value: value2 }));
let clearSearch = useEvent(() => dispatch({ type: 4 /* ClearSearch */ }));
let actions = (0, import_react34.useMemo)(
() => ({
onChange,
registerOption,
registerLabel,
goToOption,
closeListbox,
openListbox,
selectActiveOption,
selectOption,
search,
clearSearch
}),
[]
);
let ourProps = { ref: listboxRef };
let form = (0, import_react34.useRef)(null);
let d = useDisposables();
(0, import_react34.useEffect)(() => {
if (!form.current)
return;
if (defaultValue === void 0)
return;
d.addEventListener(form.current, "reset", () => {
theirOnChange == null ? void 0 : theirOnChange(defaultValue);
});
}, [
form,
theirOnChange
/* Explicitly ignoring `defaultValue` */
]);
return /* @__PURE__ */ import_react34.default.createElement(ListboxActionsContext.Provider, { value: actions }, /* @__PURE__ */ import_react34.default.createElement(ListboxDataContext.Provider, { value: data }, /* @__PURE__ */ import_react34.default.createElement(
OpenClosedProvider,
{
value: match(data.listboxState, {
[0 /* Open */]: 1 /* Open */,
[1 /* Closed */]: 2 /* Closed */
})
},
name != null && value != null && objectToFormEntries({ [name]: value }).map(([name2, value2], idx) => /* @__PURE__ */ import_react34.default.createElement(
Hidden,
{
features: 4 /* Hidden */,
ref: idx === 0 ? (element) => {
var _a3;
form.current = (_a3 = element == null ? void 0 : element.closest("form")) != null ? _a3 : null;
} : void 0,
...compact({
key: name2,
as: "input",
type: "hidden",
hidden: true,
readOnly: true,
form: formName,
disabled,
name: name2,
value: value2
})
}
)),
render({ ourProps, theirProps, slot, defaultTag: DEFAULT_LISTBOX_TAG, name: "Listbox" })
)));
}
var DEFAULT_BUTTON_TAG3 = "button";
function ButtonFn3(props, ref) {
var _a3;
let internalId = useId();
let { id = `headlessui-listbox-button-${internalId}`, ...theirProps } = props;
let data = useData2("Listbox.Button");
let actions = useActions2("Listbox.Button");
let buttonRef = useSyncRefs(data.buttonRef, ref);
let d = useDisposables();
let handleKeyDown = useEvent((event) => {
switch (event.key) {
case " " /* Space */:
case "Enter" /* Enter */:
case "ArrowDown" /* ArrowDown */:
event.preventDefault();
actions.openListbox();
d.nextFrame(() => {
if (!data.value)
actions.goToOption(0 /* First */);
});
break;
case "ArrowUp" /* ArrowUp */:
event.preventDefault();
actions.openListbox();
d.nextFrame(() => {
if (!data.value)
actions.goToOption(3 /* Last */);
});
break;
}
});
let handleKeyUp = useEvent((event) => {
switch (event.key) {
case " " /* Space */:
event.preventDefault();
break;
}
});
let handleClick = useEvent((event) => {
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
if (data.listboxState === 0 /* Open */) {
actions.closeListbox();
d.nextFrame(() => {
var _a4;
return (_a4 = data.buttonRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
} else {
event.preventDefault();
actions.openListbox();
}
});
let labelledby = useComputed(() => {
if (!data.labelId)
return void 0;
return [data.labelId, id].join(" ");
}, [data.labelId, id]);
let slot = (0, import_react34.useMemo)(
() => ({
open: data.listboxState === 0 /* Open */,
disabled: data.disabled,
value: data.value
}),
[data]
);
let ourProps = {
ref: buttonRef,
id,
type: useResolveButtonType(props, data.buttonRef),
"aria-haspopup": "listbox",
"aria-controls": (_a3 = data.optionsRef.current) == null ? void 0 : _a3.id,
"aria-expanded": data.listboxState === 0 /* Open */,
"aria-labelledby": labelledby,
disabled: data.disabled,
onKeyDown: handleKeyDown,
onKeyUp: handleKeyUp,
onClick: handleClick
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_BUTTON_TAG3,
name: "Listbox.Button"
});
}
var DEFAULT_LABEL_TAG2 = "label";
function LabelFn2(props, ref) {
let internalId = useId();
let { id = `headlessui-listbox-label-${internalId}`, ...theirProps } = props;
let data = useData2("Listbox.Label");
let actions = useActions2("Listbox.Label");
let labelRef = useSyncRefs(data.labelRef, ref);
useIsoMorphicEffect(() => actions.registerLabel(id), [id]);
let handleClick = useEvent(() => {
var _a3;
return (_a3 = data.buttonRef.current) == null ? void 0 : _a3.focus({ preventScroll: true });
});
let slot = (0, import_react34.useMemo)(
() => ({ open: data.listboxState === 0 /* Open */, disabled: data.disabled }),
[data]
);
let ourProps = { ref: labelRef, id, onClick: handleClick };
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_LABEL_TAG2,
name: "Listbox.Label"
});
}
var DEFAULT_OPTIONS_TAG2 = "ul";
var OptionsRenderFeatures2 = 1 /* RenderStrategy */ | 2 /* Static */;
function OptionsFn2(props, ref) {
var _a3;
let internalId = useId();
let { id = `headlessui-listbox-options-${internalId}`, ...theirProps } = props;
let data = useData2("Listbox.Options");
let actions = useActions2("Listbox.Options");
let optionsRef = useSyncRefs(data.optionsRef, ref);
let d = useDisposables();
let searchDisposables = useDisposables();
let usesOpenClosedState = useOpenClosed();
let visible = (() => {
if (usesOpenClosedState !== null) {
return (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
return data.listboxState === 0 /* Open */;
})();
(0, import_react34.useEffect)(() => {
var _a4;
let container = data.optionsRef.current;
if (!container)
return;
if (data.listboxState !== 0 /* Open */)
return;
if (container === ((_a4 = getOwnerDocument(container)) == null ? void 0 : _a4.activeElement))
return;
container.focus({ preventScroll: true });
}, [data.listboxState, data.optionsRef]);
let handleKeyDown = useEvent((event) => {
searchDisposables.dispose();
switch (event.key) {
case " " /* Space */:
if (data.searchQuery !== "") {
event.preventDefault();
event.stopPropagation();
return actions.search(event.key);
}
case "Enter" /* Enter */:
event.preventDefault();
event.stopPropagation();
if (data.activeOptionIndex !== null) {
let { dataRef } = data.options[data.activeOptionIndex];
actions.onChange(dataRef.current.value);
}
if (data.mode === 0 /* Single */) {
actions.closeListbox();
disposables().nextFrame(() => {
var _a4;
return (_a4 = data.buttonRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
}
break;
case match(data.orientation, { vertical: "ArrowDown" /* ArrowDown */, horizontal: "ArrowRight" /* ArrowRight */ }):
event.preventDefault();
event.stopPropagation();
return actions.goToOption(2 /* Next */);
case match(data.orientation, { vertical: "ArrowUp" /* ArrowUp */, horizontal: "ArrowLeft" /* ArrowLeft */ }):
event.preventDefault();
event.stopPropagation();
return actions.goToOption(1 /* Previous */);
case "Home" /* Home */:
case "PageUp" /* PageUp */:
event.preventDefault();
event.stopPropagation();
return actions.goToOption(0 /* First */);
case "End" /* End */:
case "PageDown" /* PageDown */:
event.preventDefault();
event.stopPropagation();
return actions.goToOption(3 /* Last */);
case "Escape" /* Escape */:
event.preventDefault();
event.stopPropagation();
actions.closeListbox();
return d.nextFrame(() => {
var _a4;
return (_a4 = data.buttonRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
case "Tab" /* Tab */:
event.preventDefault();
event.stopPropagation();
break;
default:
if (event.key.length === 1) {
actions.search(event.key);
searchDisposables.setTimeout(() => actions.clearSearch(), 350);
}
break;
}
});
let labelledby = useComputed(() => {
var _a4;
return (_a4 = data.buttonRef.current) == null ? void 0 : _a4.id;
}, [data.buttonRef.current]);
let slot = (0, import_react34.useMemo)(
() => ({ open: data.listboxState === 0 /* Open */ }),
[data]
);
let ourProps = {
"aria-activedescendant": data.activeOptionIndex === null ? void 0 : (_a3 = data.options[data.activeOptionIndex]) == null ? void 0 : _a3.id,
"aria-multiselectable": data.mode === 1 /* Multi */ ? true : void 0,
"aria-labelledby": labelledby,
"aria-orientation": data.orientation,
id,
onKeyDown: handleKeyDown,
role: "listbox",
tabIndex: 0,
ref: optionsRef
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OPTIONS_TAG2,
features: OptionsRenderFeatures2,
visible,
name: "Listbox.Options"
});
}
var DEFAULT_OPTION_TAG2 = "li";
function OptionFn2(props, ref) {
let internalId = useId();
let {
id = `headlessui-listbox-option-${internalId}`,
disabled = false,
value,
...theirProps
} = props;
let data = useData2("Listbox.Option");
let actions = useActions2("Listbox.Option");
let active = data.activeOptionIndex !== null ? data.options[data.activeOptionIndex].id === id : false;
let selected = data.isSelected(value);
let internalOptionRef = (0, import_react34.useRef)(null);
let getTextValue2 = useTextValue(internalOptionRef);
let bag = useLatestValue({
disabled,
value,
domRef: internalOptionRef,
get textValue() {
return getTextValue2();
}
});
let optionRef = useSyncRefs(ref, internalOptionRef);
useIsoMorphicEffect(() => {
if (data.listboxState !== 0 /* Open */)
return;
if (!active)
return;
if (data.activationTrigger === 0 /* Pointer */)
return;
let d = disposables();
d.requestAnimationFrame(() => {
var _a3, _b;
(_b = (_a3 = internalOptionRef.current) == null ? void 0 : _a3.scrollIntoView) == null ? void 0 : _b.call(_a3, { block: "nearest" });
});
return d.dispose;
}, [
internalOptionRef,
active,
data.listboxState,
data.activationTrigger,
/* We also want to trigger this when the position of the active item changes so that we can re-trigger the scrollIntoView */
data.activeOptionIndex
]);
useIsoMorphicEffect(() => actions.registerOption(id, bag), [bag, id]);
let handleClick = useEvent((event) => {
if (disabled)
return event.preventDefault();
actions.onChange(value);
if (data.mode === 0 /* Single */) {
actions.closeListbox();
disposables().nextFrame(() => {
var _a3;
return (_a3 = data.buttonRef.current) == null ? void 0 : _a3.focus({ preventScroll: true });
});
}
});
let handleFocus = useEvent(() => {
if (disabled)
return actions.goToOption(5 /* Nothing */);
actions.goToOption(4 /* Specific */, id);
});
let pointer = useTrackedPointer();
let handleEnter = useEvent((evt) => pointer.update(evt));
let handleMove = useEvent((evt) => {
if (!pointer.wasMoved(evt))
return;
if (disabled)
return;
if (active)
return;
actions.goToOption(4 /* Specific */, id, 0 /* Pointer */);
});
let handleLeave = useEvent((evt) => {
if (!pointer.wasMoved(evt))
return;
if (disabled)
return;
if (!active)
return;
actions.goToOption(5 /* Nothing */);
});
let slot = (0, import_react34.useMemo)(
() => ({ active, selected, disabled }),
[active, selected, disabled]
);
let ourProps = {
id,
ref: optionRef,
role: "option",
tabIndex: disabled === true ? void 0 : -1,
"aria-disabled": disabled === true ? true : void 0,
// According to the WAI-ARIA best practices, we should use aria-checked for
// multi-select,but Voice-Over disagrees. So we use aria-checked instead for
// both single and multi-select.
"aria-selected": selected,
disabled: void 0,
// Never forward the `disabled` prop
onClick: handleClick,
onFocus: handleFocus,
onPointerEnter: handleEnter,
onMouseEnter: handleEnter,
onPointerMove: handleMove,
onMouseMove: handleMove,
onPointerLeave: handleLeave,
onMouseLeave: handleLeave
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OPTION_TAG2,
name: "Listbox.Option"
});
}
var ListboxRoot = forwardRefWithAs(ListboxFn);
var Button3 = forwardRefWithAs(ButtonFn3);
var Label2 = forwardRefWithAs(LabelFn2);
var Options2 = forwardRefWithAs(OptionsFn2);
var Option2 = forwardRefWithAs(OptionFn2);
var Listbox = Object.assign(ListboxRoot, { Button: Button3, Label: Label2, Options: Options2, Option: Option2 });
// src/components/menu/menu.tsx
var import_react35 = __toESM(require("react"), 1);
function adjustOrderedState3(state, adjustment = (i) => i) {
let currentActiveItem = state.activeItemIndex !== null ? state.items[state.activeItemIndex] : null;
let sortedItems = sortByDomNode(
adjustment(state.items.slice()),
(item) => item.dataRef.current.domRef.current
);
let adjustedActiveItemIndex = currentActiveItem ? sortedItems.indexOf(currentActiveItem) : null;
if (adjustedActiveItemIndex === -1) {
adjustedActiveItemIndex = null;
}
return {
items: sortedItems,
activeItemIndex: adjustedActiveItemIndex
};
}
var reducers5 = {
[1 /* CloseMenu */](state) {
if (state.menuState === 1 /* Closed */)
return state;
return { ...state, activeItemIndex: null, menuState: 1 /* Closed */ };
},
[0 /* OpenMenu */](state) {
if (state.menuState === 0 /* Open */)
return state;
return {
...state,
/* We can turn off demo mode once we re-open the `Menu` */
__demoMode: false,
menuState: 0 /* Open */
};
},
[2 /* GoToItem */]: (state, action) => {
var _a3;
let adjustedState = adjustOrderedState3(state);
let activeItemIndex = calculateActiveIndex(action, {
resolveItems: () => adjustedState.items,
resolveActiveIndex: () => adjustedState.activeItemIndex,
resolveId: (item) => item.id,
resolveDisabled: (item) => item.dataRef.current.disabled
});
return {
...state,
...adjustedState,
searchQuery: "",
activeItemIndex,
activationTrigger: (_a3 = action.trigger) != null ? _a3 : 1 /* Other */
};
},
[3 /* Search */]: (state, action) => {
let wasAlreadySearching = state.searchQuery !== "";
let offset = wasAlreadySearching ? 0 : 1;
let searchQuery = state.searchQuery + action.value.toLowerCase();
let reOrderedItems = state.activeItemIndex !== null ? state.items.slice(state.activeItemIndex + offset).concat(state.items.slice(0, state.activeItemIndex + offset)) : state.items;
let matchingItem = reOrderedItems.find(
(item) => {
var _a3;
return ((_a3 = item.dataRef.current.textValue) == null ? void 0 : _a3.startsWith(searchQuery)) && !item.dataRef.current.disabled;
}
);
let matchIdx = matchingItem ? state.items.indexOf(matchingItem) : -1;
if (matchIdx === -1 || matchIdx === state.activeItemIndex)
return { ...state, searchQuery };
return {
...state,
searchQuery,
activeItemIndex: matchIdx,
activationTrigger: 1 /* Other */
};
},
[4 /* ClearSearch */](state) {
if (state.searchQuery === "")
return state;
return { ...state, searchQuery: "", searchActiveItemIndex: null };
},
[5 /* RegisterItem */]: (state, action) => {
let adjustedState = adjustOrderedState3(state, (items) => [
...items,
{ id: action.id, dataRef: action.dataRef }
]);
return { ...state, ...adjustedState };
},
[6 /* UnregisterItem */]: (state, action) => {
let adjustedState = adjustOrderedState3(state, (items) => {
let idx = items.findIndex((a) => a.id === action.id);
if (idx !== -1)
items.splice(idx, 1);
return items;
});
return {
...state,
...adjustedState,
activationTrigger: 1 /* Other */
};
}
};
var MenuContext = (0, import_react35.createContext)(null);
MenuContext.displayName = "MenuContext";
function useMenuContext(component) {
let context = (0, import_react35.useContext)(MenuContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Menu /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useMenuContext);
throw err;
}
return context;
}
function stateReducer5(state, action) {
return match(action.type, reducers5, state, action);
}
var DEFAULT_MENU_TAG = import_react35.Fragment;
function MenuFn(props, ref) {
let { __demoMode = false, ...theirProps } = props;
let reducerBag = (0, import_react35.useReducer)(stateReducer5, {
__demoMode,
menuState: __demoMode ? 0 /* Open */ : 1 /* Closed */,
buttonRef: (0, import_react35.createRef)(),
itemsRef: (0, import_react35.createRef)(),
items: [],
searchQuery: "",
activeItemIndex: null,
activationTrigger: 1 /* Other */
});
let [{ menuState, itemsRef, buttonRef }, dispatch] = reducerBag;
let menuRef = useSyncRefs(ref);
useOutsideClick(
[buttonRef, itemsRef],
(event, target) => {
var _a3;
dispatch({ type: 1 /* CloseMenu */ });
if (!isFocusableElement(target, 1 /* Loose */)) {
event.preventDefault();
(_a3 = buttonRef.current) == null ? void 0 : _a3.focus();
}
},
menuState === 0 /* Open */
);
let close = useEvent(() => {
dispatch({ type: 1 /* CloseMenu */ });
});
let slot = (0, import_react35.useMemo)(
() => ({ open: menuState === 0 /* Open */, close }),
[menuState, close]
);
let ourProps = { ref: menuRef };
return /* @__PURE__ */ import_react35.default.createElement(MenuContext.Provider, { value: reducerBag }, /* @__PURE__ */ import_react35.default.createElement(
OpenClosedProvider,
{
value: match(menuState, {
[0 /* Open */]: 1 /* Open */,
[1 /* Closed */]: 2 /* Closed */
})
},
render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_MENU_TAG,
name: "Menu"
})
));
}
var DEFAULT_BUTTON_TAG4 = "button";
function ButtonFn4(props, ref) {
var _a3;
let internalId = useId();
let { id = `headlessui-menu-button-${internalId}`, ...theirProps } = props;
let [state, dispatch] = useMenuContext("Menu.Button");
let buttonRef = useSyncRefs(state.buttonRef, ref);
let d = useDisposables();
let handleKeyDown = useEvent((event) => {
switch (event.key) {
case " " /* Space */:
case "Enter" /* Enter */:
case "ArrowDown" /* ArrowDown */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 0 /* OpenMenu */ });
d.nextFrame(() => dispatch({ type: 2 /* GoToItem */, focus: 0 /* First */ }));
break;
case "ArrowUp" /* ArrowUp */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 0 /* OpenMenu */ });
d.nextFrame(() => dispatch({ type: 2 /* GoToItem */, focus: 3 /* Last */ }));
break;
}
});
let handleKeyUp = useEvent((event) => {
switch (event.key) {
case " " /* Space */:
event.preventDefault();
break;
}
});
let handleClick = useEvent((event) => {
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
if (props.disabled)
return;
if (state.menuState === 0 /* Open */) {
dispatch({ type: 1 /* CloseMenu */ });
d.nextFrame(() => {
var _a4;
return (_a4 = state.buttonRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
});
} else {
event.preventDefault();
dispatch({ type: 0 /* OpenMenu */ });
}
});
let slot = (0, import_react35.useMemo)(
() => ({ open: state.menuState === 0 /* Open */ }),
[state]
);
let ourProps = {
ref: buttonRef,
id,
type: useResolveButtonType(props, state.buttonRef),
"aria-haspopup": "menu",
"aria-controls": (_a3 = state.itemsRef.current) == null ? void 0 : _a3.id,
"aria-expanded": state.menuState === 0 /* Open */,
onKeyDown: handleKeyDown,
onKeyUp: handleKeyUp,
onClick: handleClick
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_BUTTON_TAG4,
name: "Menu.Button"
});
}
var DEFAULT_ITEMS_TAG = "div";
var ItemsRenderFeatures = 1 /* RenderStrategy */ | 2 /* Static */;
function ItemsFn(props, ref) {
var _a3, _b;
let internalId = useId();
let { id = `headlessui-menu-items-${internalId}`, ...theirProps } = props;
let [state, dispatch] = useMenuContext("Menu.Items");
let itemsRef = useSyncRefs(state.itemsRef, ref);
let ownerDocument = useOwnerDocument(state.itemsRef);
let searchDisposables = useDisposables();
let usesOpenClosedState = useOpenClosed();
let visible = (() => {
if (usesOpenClosedState !== null) {
return (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
return state.menuState === 0 /* Open */;
})();
(0, import_react35.useEffect)(() => {
let container = state.itemsRef.current;
if (!container)
return;
if (state.menuState !== 0 /* Open */)
return;
if (container === (ownerDocument == null ? void 0 : ownerDocument.activeElement))
return;
container.focus({ preventScroll: true });
}, [state.menuState, state.itemsRef, ownerDocument]);
useTreeWalker({
container: state.itemsRef.current,
enabled: state.menuState === 0 /* Open */,
accept(node) {
if (node.getAttribute("role") === "menuitem")
return NodeFilter.FILTER_REJECT;
if (node.hasAttribute("role"))
return NodeFilter.FILTER_SKIP;
return NodeFilter.FILTER_ACCEPT;
},
walk(node) {
node.setAttribute("role", "none");
}
});
let handleKeyDown = useEvent((event) => {
var _a4, _b2;
searchDisposables.dispose();
switch (event.key) {
case " " /* Space */:
if (state.searchQuery !== "") {
event.preventDefault();
event.stopPropagation();
return dispatch({ type: 3 /* Search */, value: event.key });
}
case "Enter" /* Enter */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 1 /* CloseMenu */ });
if (state.activeItemIndex !== null) {
let { dataRef } = state.items[state.activeItemIndex];
(_b2 = (_a4 = dataRef.current) == null ? void 0 : _a4.domRef.current) == null ? void 0 : _b2.click();
}
restoreFocusIfNecessary(state.buttonRef.current);
break;
case "ArrowDown" /* ArrowDown */:
event.preventDefault();
event.stopPropagation();
return dispatch({ type: 2 /* GoToItem */, focus: 2 /* Next */ });
case "ArrowUp" /* ArrowUp */:
event.preventDefault();
event.stopPropagation();
return dispatch({ type: 2 /* GoToItem */, focus: 1 /* Previous */ });
case "Home" /* Home */:
case "PageUp" /* PageUp */:
event.preventDefault();
event.stopPropagation();
return dispatch({ type: 2 /* GoToItem */, focus: 0 /* First */ });
case "End" /* End */:
case "PageDown" /* PageDown */:
event.preventDefault();
event.stopPropagation();
return dispatch({ type: 2 /* GoToItem */, focus: 3 /* Last */ });
case "Escape" /* Escape */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 1 /* CloseMenu */ });
disposables().nextFrame(() => {
var _a5;
return (_a5 = state.buttonRef.current) == null ? void 0 : _a5.focus({ preventScroll: true });
});
break;
case "Tab" /* Tab */:
event.preventDefault();
event.stopPropagation();
dispatch({ type: 1 /* CloseMenu */ });
disposables().nextFrame(() => {
focusFrom(
state.buttonRef.current,
event.shiftKey ? 2 /* Previous */ : 4 /* Next */
);
});
break;
default:
if (event.key.length === 1) {
dispatch({ type: 3 /* Search */, value: event.key });
searchDisposables.setTimeout(() => dispatch({ type: 4 /* ClearSearch */ }), 350);
}
break;
}
});
let handleKeyUp = useEvent((event) => {
switch (event.key) {
case " " /* Space */:
event.preventDefault();
break;
}
});
let slot = (0, import_react35.useMemo)(
() => ({ open: state.menuState === 0 /* Open */ }),
[state]
);
let ourProps = {
"aria-activedescendant": state.activeItemIndex === null ? void 0 : (_a3 = state.items[state.activeItemIndex]) == null ? void 0 : _a3.id,
"aria-labelledby": (_b = state.buttonRef.current) == null ? void 0 : _b.id,
id,
onKeyDown: handleKeyDown,
onKeyUp: handleKeyUp,
role: "menu",
tabIndex: 0,
ref: itemsRef
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_ITEMS_TAG,
features: ItemsRenderFeatures,
visible,
name: "Menu.Items"
});
}
var DEFAULT_ITEM_TAG = import_react35.Fragment;
function ItemFn(props, ref) {
let internalId = useId();
let { id = `headlessui-menu-item-${internalId}`, disabled = false, ...theirProps } = props;
let [state, dispatch] = useMenuContext("Menu.Item");
let active = state.activeItemIndex !== null ? state.items[state.activeItemIndex].id === id : false;
let internalItemRef = (0, import_react35.useRef)(null);
let itemRef = useSyncRefs(ref, internalItemRef);
useIsoMorphicEffect(() => {
if (state.__demoMode)
return;
if (state.menuState !== 0 /* Open */)
return;
if (!active)
return;
if (state.activationTrigger === 0 /* Pointer */)
return;
let d = disposables();
d.requestAnimationFrame(() => {
var _a3, _b;
(_b = (_a3 = internalItemRef.current) == null ? void 0 : _a3.scrollIntoView) == null ? void 0 : _b.call(_a3, { block: "nearest" });
});
return d.dispose;
}, [
state.__demoMode,
internalItemRef,
active,
state.menuState,
state.activationTrigger,
/* We also want to trigger this when the position of the active item changes so that we can re-trigger the scrollIntoView */
state.activeItemIndex
]);
let getTextValue2 = useTextValue(internalItemRef);
let bag = (0, import_react35.useRef)({
disabled,
domRef: internalItemRef,
get textValue() {
return getTextValue2();
}
});
useIsoMorphicEffect(() => {
bag.current.disabled = disabled;
}, [bag, disabled]);
useIsoMorphicEffect(() => {
dispatch({ type: 5 /* RegisterItem */, id, dataRef: bag });
return () => dispatch({ type: 6 /* UnregisterItem */, id });
}, [bag, id]);
let close = useEvent(() => {
dispatch({ type: 1 /* CloseMenu */ });
});
let handleClick = useEvent((event) => {
if (disabled)
return event.preventDefault();
dispatch({ type: 1 /* CloseMenu */ });
restoreFocusIfNecessary(state.buttonRef.current);
});
let handleFocus = useEvent(() => {
if (disabled)
return dispatch({ type: 2 /* GoToItem */, focus: 5 /* Nothing */ });
dispatch({ type: 2 /* GoToItem */, focus: 4 /* Specific */, id });
});
let pointer = useTrackedPointer();
let handleEnter = useEvent((evt) => pointer.update(evt));
let handleMove = useEvent((evt) => {
if (!pointer.wasMoved(evt))
return;
if (disabled)
return;
if (active)
return;
dispatch({
type: 2 /* GoToItem */,
focus: 4 /* Specific */,
id,
trigger: 0 /* Pointer */
});
});
let handleLeave = useEvent((evt) => {
if (!pointer.wasMoved(evt))
return;
if (disabled)
return;
if (!active)
return;
dispatch({ type: 2 /* GoToItem */, focus: 5 /* Nothing */ });
});
let slot = (0, import_react35.useMemo)(
() => ({ active, disabled, close }),
[active, disabled, close]
);
let ourProps = {
id,
ref: itemRef,
role: "menuitem",
tabIndex: disabled === true ? void 0 : -1,
"aria-disabled": disabled === true ? true : void 0,
disabled: void 0,
// Never forward the `disabled` prop
onClick: handleClick,
onFocus: handleFocus,
onPointerEnter: handleEnter,
onMouseEnter: handleEnter,
onPointerMove: handleMove,
onMouseMove: handleMove,
onPointerLeave: handleLeave,
onMouseLeave: handleLeave
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_ITEM_TAG,
name: "Menu.Item"
});
}
var MenuRoot = forwardRefWithAs(MenuFn);
var Button4 = forwardRefWithAs(ButtonFn4);
var Items = forwardRefWithAs(ItemsFn);
var Item = forwardRefWithAs(ItemFn);
var Menu = Object.assign(MenuRoot, { Button: Button4, Items, Item });
// src/components/popover/popover.tsx
var import_react36 = __toESM(require("react"), 1);
var reducers6 = {
[0 /* TogglePopover */]: (state) => {
let nextState = {
...state,
popoverState: match(state.popoverState, {
[0 /* Open */]: 1 /* Closed */,
[1 /* Closed */]: 0 /* Open */
})
};
if (nextState.popoverState === 0 /* Open */) {
nextState.__demoMode = false;
}
return nextState;
},
[1 /* ClosePopover */](state) {
if (state.popoverState === 1 /* Closed */)
return state;
return { ...state, popoverState: 1 /* Closed */ };
},
[2 /* SetButton */](state, action) {
if (state.button === action.button)
return state;
return { ...state, button: action.button };
},
[3 /* SetButtonId */](state, action) {
if (state.buttonId === action.buttonId)
return state;
return { ...state, buttonId: action.buttonId };
},
[4 /* SetPanel */](state, action) {
if (state.panel === action.panel)
return state;
return { ...state, panel: action.panel };
},
[5 /* SetPanelId */](state, action) {
if (state.panelId === action.panelId)
return state;
return { ...state, panelId: action.panelId };
}
};
var PopoverContext = (0, import_react36.createContext)(null);
PopoverContext.displayName = "PopoverContext";
function usePopoverContext(component) {
let context = (0, import_react36.useContext)(PopoverContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Popover /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, usePopoverContext);
throw err;
}
return context;
}
var PopoverAPIContext = (0, import_react36.createContext)(null);
PopoverAPIContext.displayName = "PopoverAPIContext";
function usePopoverAPIContext(component) {
let context = (0, import_react36.useContext)(PopoverAPIContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Popover /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, usePopoverAPIContext);
throw err;
}
return context;
}
var PopoverGroupContext = (0, import_react36.createContext)(null);
PopoverGroupContext.displayName = "PopoverGroupContext";
function usePopoverGroupContext() {
return (0, import_react36.useContext)(PopoverGroupContext);
}
var PopoverPanelContext = (0, import_react36.createContext)(null);
PopoverPanelContext.displayName = "PopoverPanelContext";
function usePopoverPanelContext() {
return (0, import_react36.useContext)(PopoverPanelContext);
}
function stateReducer6(state, action) {
return match(action.type, reducers6, state, action);
}
var DEFAULT_POPOVER_TAG = "div";
function PopoverFn(props, ref) {
var _a3;
let { __demoMode = false, ...theirProps } = props;
let internalPopoverRef = (0, import_react36.useRef)(null);
let popoverRef = useSyncRefs(
ref,
optionalRef((ref2) => {
internalPopoverRef.current = ref2;
})
);
let buttons = (0, import_react36.useRef)([]);
let reducerBag = (0, import_react36.useReducer)(stateReducer6, {
__demoMode,
popoverState: __demoMode ? 0 /* Open */ : 1 /* Closed */,
buttons,
button: null,
buttonId: null,
panel: null,
panelId: null,
beforePanelSentinel: (0, import_react36.createRef)(),
afterPanelSentinel: (0, import_react36.createRef)()
});
let [
{ popoverState, button, buttonId, panel, panelId, beforePanelSentinel, afterPanelSentinel },
dispatch
] = reducerBag;
let ownerDocument = useOwnerDocument((_a3 = internalPopoverRef.current) != null ? _a3 : button);
let isPortalled = (0, import_react36.useMemo)(() => {
if (!button)
return false;
if (!panel)
return false;
for (let root2 of document.querySelectorAll("body > *")) {
if (Number(root2 == null ? void 0 : root2.contains(button)) ^ Number(root2 == null ? void 0 : root2.contains(panel))) {
return true;
}
}
let elements = getFocusableElements();
let buttonIdx = elements.indexOf(button);
let beforeIdx = (buttonIdx + elements.length - 1) % elements.length;
let afterIdx = (buttonIdx + 1) % elements.length;
let beforeElement = elements[beforeIdx];
let afterElement = elements[afterIdx];
if (!panel.contains(beforeElement) && !panel.contains(afterElement)) {
return true;
}
return false;
}, [button, panel]);
let buttonIdRef = useLatestValue(buttonId);
let panelIdRef = useLatestValue(panelId);
let registerBag = (0, import_react36.useMemo)(
() => ({
buttonId: buttonIdRef,
panelId: panelIdRef,
close: () => dispatch({ type: 1 /* ClosePopover */ })
}),
[buttonIdRef, panelIdRef, dispatch]
);
let groupContext = usePopoverGroupContext();
let registerPopover = groupContext == null ? void 0 : groupContext.registerPopover;
let isFocusWithinPopoverGroup = useEvent(() => {
var _a4;
return (_a4 = groupContext == null ? void 0 : groupContext.isFocusWithinPopoverGroup()) != null ? _a4 : (ownerDocument == null ? void 0 : ownerDocument.activeElement) && ((button == null ? void 0 : button.contains(ownerDocument.activeElement)) || (panel == null ? void 0 : panel.contains(ownerDocument.activeElement)));
});
(0, import_react36.useEffect)(() => registerPopover == null ? void 0 : registerPopover(registerBag), [registerPopover, registerBag]);
let [portals, PortalWrapper] = useNestedPortals();
let root = useRootContainers({
mainTreeNodeRef: groupContext == null ? void 0 : groupContext.mainTreeNodeRef,
portals,
defaultContainers: [button, panel]
});
useEventListener(
ownerDocument == null ? void 0 : ownerDocument.defaultView,
"focus",
(event) => {
var _a4, _b, _c, _d;
if (event.target === window)
return;
if (!(event.target instanceof HTMLElement))
return;
if (popoverState !== 0 /* Open */)
return;
if (isFocusWithinPopoverGroup())
return;
if (!button)
return;
if (!panel)
return;
if (root.contains(event.target))
return;
if ((_b = (_a4 = beforePanelSentinel.current) == null ? void 0 : _a4.contains) == null ? void 0 : _b.call(_a4, event.target))
return;
if ((_d = (_c = afterPanelSentinel.current) == null ? void 0 : _c.contains) == null ? void 0 : _d.call(_c, event.target))
return;
dispatch({ type: 1 /* ClosePopover */ });
},
true
);
useOutsideClick(
root.resolveContainers,
(event, target) => {
dispatch({ type: 1 /* ClosePopover */ });
if (!isFocusableElement(target, 1 /* Loose */)) {
event.preventDefault();
button == null ? void 0 : button.focus();
}
},
popoverState === 0 /* Open */
);
let close = useEvent(
(focusableElement) => {
dispatch({ type: 1 /* ClosePopover */ });
let restoreElement = (() => {
if (!focusableElement)
return button;
if (focusableElement instanceof HTMLElement)
return focusableElement;
if ("current" in focusableElement && focusableElement.current instanceof HTMLElement)
return focusableElement.current;
return button;
})();
restoreElement == null ? void 0 : restoreElement.focus();
}
);
let api = (0, import_react36.useMemo)(
() => ({ close, isPortalled }),
[close, isPortalled]
);
let slot = (0, import_react36.useMemo)(
() => ({ open: popoverState === 0 /* Open */, close }),
[popoverState, close]
);
let ourProps = { ref: popoverRef };
return /* @__PURE__ */ import_react36.default.createElement(PopoverPanelContext.Provider, { value: null }, /* @__PURE__ */ import_react36.default.createElement(PopoverContext.Provider, { value: reducerBag }, /* @__PURE__ */ import_react36.default.createElement(PopoverAPIContext.Provider, { value: api }, /* @__PURE__ */ import_react36.default.createElement(
OpenClosedProvider,
{
value: match(popoverState, {
[0 /* Open */]: 1 /* Open */,
[1 /* Closed */]: 2 /* Closed */
})
},
/* @__PURE__ */ import_react36.default.createElement(PortalWrapper, null, render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_POPOVER_TAG,
name: "Popover"
}), /* @__PURE__ */ import_react36.default.createElement(root.MainTreeNode, null))
))));
}
var DEFAULT_BUTTON_TAG5 = "button";
function ButtonFn5(props, ref) {
let internalId = useId();
let { id = `headlessui-popover-button-${internalId}`, ...theirProps } = props;
let [state, dispatch] = usePopoverContext("Popover.Button");
let { isPortalled } = usePopoverAPIContext("Popover.Button");
let internalButtonRef = (0, import_react36.useRef)(null);
let sentinelId = `headlessui-focus-sentinel-${useId()}`;
let groupContext = usePopoverGroupContext();
let closeOthers = groupContext == null ? void 0 : groupContext.closeOthers;
let panelContext = usePopoverPanelContext();
let isWithinPanel = panelContext !== null;
(0, import_react36.useEffect)(() => {
if (isWithinPanel)
return;
dispatch({ type: 3 /* SetButtonId */, buttonId: id });
return () => {
dispatch({ type: 3 /* SetButtonId */, buttonId: null });
};
}, [isWithinPanel, id, dispatch]);
let [uniqueIdentifier] = (0, import_react36.useState)(() => Symbol());
let buttonRef = useSyncRefs(
internalButtonRef,
ref,
isWithinPanel ? null : (button) => {
if (button) {
state.buttons.current.push(uniqueIdentifier);
} else {
let idx = state.buttons.current.indexOf(uniqueIdentifier);
if (idx !== -1)
state.buttons.current.splice(idx, 1);
}
if (state.buttons.current.length > 1) {
console.warn(
"You are already using a <Popover.Button /> but only 1 <Popover.Button /> is supported."
);
}
button && dispatch({ type: 2 /* SetButton */, button });
}
);
let withinPanelButtonRef = useSyncRefs(internalButtonRef, ref);
let ownerDocument = useOwnerDocument(internalButtonRef);
let handleKeyDown = useEvent((event) => {
var _a3, _b, _c;
if (isWithinPanel) {
if (state.popoverState === 1 /* Closed */)
return;
switch (event.key) {
case " " /* Space */:
case "Enter" /* Enter */:
event.preventDefault();
(_b = (_a3 = event.target).click) == null ? void 0 : _b.call(_a3);
dispatch({ type: 1 /* ClosePopover */ });
(_c = state.button) == null ? void 0 : _c.focus();
break;
}
} else {
switch (event.key) {
case " " /* Space */:
case "Enter" /* Enter */:
event.preventDefault();
event.stopPropagation();
if (state.popoverState === 1 /* Closed */)
closeOthers == null ? void 0 : closeOthers(state.buttonId);
dispatch({ type: 0 /* TogglePopover */ });
break;
case "Escape" /* Escape */:
if (state.popoverState !== 0 /* Open */)
return closeOthers == null ? void 0 : closeOthers(state.buttonId);
if (!internalButtonRef.current)
return;
if ((ownerDocument == null ? void 0 : ownerDocument.activeElement) && !internalButtonRef.current.contains(ownerDocument.activeElement)) {
return;
}
event.preventDefault();
event.stopPropagation();
dispatch({ type: 1 /* ClosePopover */ });
break;
}
}
});
let handleKeyUp = useEvent((event) => {
if (isWithinPanel)
return;
if (event.key === " " /* Space */) {
event.preventDefault();
}
});
let handleClick = useEvent((event) => {
var _a3, _b;
if (isDisabledReactIssue7711(event.currentTarget))
return;
if (props.disabled)
return;
if (isWithinPanel) {
dispatch({ type: 1 /* ClosePopover */ });
(_a3 = state.button) == null ? void 0 : _a3.focus();
} else {
event.preventDefault();
event.stopPropagation();
if (state.popoverState === 1 /* Closed */)
closeOthers == null ? void 0 : closeOthers(state.buttonId);
dispatch({ type: 0 /* TogglePopover */ });
(_b = state.button) == null ? void 0 : _b.focus();
}
});
let handleMouseDown = useEvent((event) => {
event.preventDefault();
event.stopPropagation();
});
let visible = state.popoverState === 0 /* Open */;
let slot = (0, import_react36.useMemo)(() => ({ open: visible }), [visible]);
let type = useResolveButtonType(props, internalButtonRef);
let ourProps = isWithinPanel ? {
ref: withinPanelButtonRef,
type,
onKeyDown: handleKeyDown,
onClick: handleClick
} : {
ref: buttonRef,
id: state.buttonId,
type,
"aria-expanded": state.popoverState === 0 /* Open */,
"aria-controls": state.panel ? state.panelId : void 0,
onKeyDown: handleKeyDown,
onKeyUp: handleKeyUp,
onClick: handleClick,
onMouseDown: handleMouseDown
};
let direction = useTabDirection();
let handleFocus = useEvent(() => {
let el = state.panel;
if (!el)
return;
function run() {
let result = match(direction.current, {
[0 /* Forwards */]: () => focusIn(el, 1 /* First */),
[1 /* Backwards */]: () => focusIn(el, 8 /* Last */)
});
if (result === 0 /* Error */) {
focusIn(
getFocusableElements().filter((el2) => el2.dataset.headlessuiFocusGuard !== "true"),
match(direction.current, {
[0 /* Forwards */]: 4 /* Next */,
[1 /* Backwards */]: 2 /* Previous */
}),
{ relativeTo: state.button }
);
}
}
if (false) {
microTask(run);
} else {
run();
}
});
return /* @__PURE__ */ import_react36.default.createElement(import_react36.default.Fragment, null, render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_BUTTON_TAG5,
name: "Popover.Button"
}), visible && !isWithinPanel && isPortalled && /* @__PURE__ */ import_react36.default.createElement(
Hidden,
{
id: sentinelId,
features: 2 /* Focusable */,
"data-headlessui-focus-guard": true,
as: "button",
type: "button",
onFocus: handleFocus
}
));
}
var DEFAULT_OVERLAY_TAG2 = "div";
var OverlayRenderFeatures = 1 /* RenderStrategy */ | 2 /* Static */;
function OverlayFn2(props, ref) {
let internalId = useId();
let { id = `headlessui-popover-overlay-${internalId}`, ...theirProps } = props;
let [{ popoverState }, dispatch] = usePopoverContext("Popover.Overlay");
let overlayRef = useSyncRefs(ref);
let usesOpenClosedState = useOpenClosed();
let visible = (() => {
if (usesOpenClosedState !== null) {
return (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
return popoverState === 0 /* Open */;
})();
let handleClick = useEvent((event) => {
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
dispatch({ type: 1 /* ClosePopover */ });
});
let slot = (0, import_react36.useMemo)(
() => ({ open: popoverState === 0 /* Open */ }),
[popoverState]
);
let ourProps = {
ref: overlayRef,
id,
"aria-hidden": true,
onClick: handleClick
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OVERLAY_TAG2,
features: OverlayRenderFeatures,
visible,
name: "Popover.Overlay"
});
}
var DEFAULT_PANEL_TAG3 = "div";
var PanelRenderFeatures2 = 1 /* RenderStrategy */ | 2 /* Static */;
function PanelFn3(props, ref) {
let internalId = useId();
let { id = `headlessui-popover-panel-${internalId}`, focus = false, ...theirProps } = props;
let [state, dispatch] = usePopoverContext("Popover.Panel");
let { close, isPortalled } = usePopoverAPIContext("Popover.Panel");
let beforePanelSentinelId = `headlessui-focus-sentinel-before-${useId()}`;
let afterPanelSentinelId = `headlessui-focus-sentinel-after-${useId()}`;
let internalPanelRef = (0, import_react36.useRef)(null);
let panelRef = useSyncRefs(internalPanelRef, ref, (panel) => {
dispatch({ type: 4 /* SetPanel */, panel });
});
let ownerDocument = useOwnerDocument(internalPanelRef);
let mergeRefs = useMergeRefsFn();
useIsoMorphicEffect(() => {
dispatch({ type: 5 /* SetPanelId */, panelId: id });
return () => {
dispatch({ type: 5 /* SetPanelId */, panelId: null });
};
}, [id, dispatch]);
let usesOpenClosedState = useOpenClosed();
let visible = (() => {
if (usesOpenClosedState !== null) {
return (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
return state.popoverState === 0 /* Open */;
})();
let handleKeyDown = useEvent((event) => {
var _a3;
switch (event.key) {
case "Escape" /* Escape */:
if (state.popoverState !== 0 /* Open */)
return;
if (!internalPanelRef.current)
return;
if ((ownerDocument == null ? void 0 : ownerDocument.activeElement) && !internalPanelRef.current.contains(ownerDocument.activeElement)) {
return;
}
event.preventDefault();
event.stopPropagation();
dispatch({ type: 1 /* ClosePopover */ });
(_a3 = state.button) == null ? void 0 : _a3.focus();
break;
}
});
(0, import_react36.useEffect)(() => {
var _a3;
if (props.static)
return;
if (state.popoverState === 1 /* Closed */ && ((_a3 = props.unmount) != null ? _a3 : true)) {
dispatch({ type: 4 /* SetPanel */, panel: null });
}
}, [state.popoverState, props.unmount, props.static, dispatch]);
(0, import_react36.useEffect)(() => {
if (state.__demoMode)
return;
if (!focus)
return;
if (state.popoverState !== 0 /* Open */)
return;
if (!internalPanelRef.current)
return;
let activeElement = ownerDocument == null ? void 0 : ownerDocument.activeElement;
if (internalPanelRef.current.contains(activeElement))
return;
focusIn(internalPanelRef.current, 1 /* First */);
}, [state.__demoMode, focus, internalPanelRef, state.popoverState]);
let slot = (0, import_react36.useMemo)(
() => ({ open: state.popoverState === 0 /* Open */, close }),
[state, close]
);
let ourProps = {
ref: panelRef,
id,
onKeyDown: handleKeyDown,
onBlur: focus && state.popoverState === 0 /* Open */ ? (event) => {
var _a3, _b, _c, _d, _e;
let el = event.relatedTarget;
if (!el)
return;
if (!internalPanelRef.current)
return;
if ((_a3 = internalPanelRef.current) == null ? void 0 : _a3.contains(el))
return;
dispatch({ type: 1 /* ClosePopover */ });
if (((_c = (_b = state.beforePanelSentinel.current) == null ? void 0 : _b.contains) == null ? void 0 : _c.call(_b, el)) || ((_e = (_d = state.afterPanelSentinel.current) == null ? void 0 : _d.contains) == null ? void 0 : _e.call(_d, el))) {
el.focus({ preventScroll: true });
}
} : void 0,
tabIndex: -1
};
let direction = useTabDirection();
let handleBeforeFocus = useEvent(() => {
let el = internalPanelRef.current;
if (!el)
return;
function run() {
match(direction.current, {
[0 /* Forwards */]: () => {
var _a3;
let result = focusIn(el, 1 /* First */);
if (result === 0 /* Error */) {
(_a3 = state.afterPanelSentinel.current) == null ? void 0 : _a3.focus();
}
},
[1 /* Backwards */]: () => {
var _a3;
(_a3 = state.button) == null ? void 0 : _a3.focus({ preventScroll: true });
}
});
}
if (false) {
microTask(run);
} else {
run();
}
});
let handleAfterFocus = useEvent(() => {
let el = internalPanelRef.current;
if (!el)
return;
function run() {
match(direction.current, {
[0 /* Forwards */]: () => {
var _a3;
if (!state.button)
return;
let elements = getFocusableElements();
let idx = elements.indexOf(state.button);
let before = elements.slice(0, idx + 1);
let after = elements.slice(idx + 1);
let combined = [...after, ...before];
for (let element of combined.slice()) {
if (element.dataset.headlessuiFocusGuard === "true" || ((_a3 = state.panel) == null ? void 0 : _a3.contains(element))) {
let idx2 = combined.indexOf(element);
if (idx2 !== -1)
combined.splice(idx2, 1);
}
}
focusIn(combined, 1 /* First */, { sorted: false });
},
[1 /* Backwards */]: () => {
var _a3;
let result = focusIn(el, 2 /* Previous */);
if (result === 0 /* Error */) {
(_a3 = state.button) == null ? void 0 : _a3.focus();
}
}
});
}
if (false) {
microTask(run);
} else {
run();
}
});
return /* @__PURE__ */ import_react36.default.createElement(PopoverPanelContext.Provider, { value: id }, visible && isPortalled && /* @__PURE__ */ import_react36.default.createElement(
Hidden,
{
id: beforePanelSentinelId,
ref: state.beforePanelSentinel,
features: 2 /* Focusable */,
"data-headlessui-focus-guard": true,
as: "button",
type: "button",
onFocus: handleBeforeFocus
}
), render({
mergeRefs,
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_PANEL_TAG3,
features: PanelRenderFeatures2,
visible,
name: "Popover.Panel"
}), visible && isPortalled && /* @__PURE__ */ import_react36.default.createElement(
Hidden,
{
id: afterPanelSentinelId,
ref: state.afterPanelSentinel,
features: 2 /* Focusable */,
"data-headlessui-focus-guard": true,
as: "button",
type: "button",
onFocus: handleAfterFocus
}
));
}
var DEFAULT_GROUP_TAG2 = "div";
function GroupFn2(props, ref) {
let internalGroupRef = (0, import_react36.useRef)(null);
let groupRef = useSyncRefs(internalGroupRef, ref);
let [popovers, setPopovers] = (0, import_react36.useState)([]);
let root = useMainTreeNode();
let unregisterPopover = useEvent((registerbag) => {
setPopovers((existing) => {
let idx = existing.indexOf(registerbag);
if (idx !== -1) {
let clone = existing.slice();
clone.splice(idx, 1);
return clone;
}
return existing;
});
});
let registerPopover = useEvent((registerbag) => {
setPopovers((existing) => [...existing, registerbag]);
return () => unregisterPopover(registerbag);
});
let isFocusWithinPopoverGroup = useEvent(() => {
var _a3;
let ownerDocument = getOwnerDocument(internalGroupRef);
if (!ownerDocument)
return false;
let element = ownerDocument.activeElement;
if ((_a3 = internalGroupRef.current) == null ? void 0 : _a3.contains(element))
return true;
return popovers.some((bag) => {
var _a4, _b;
return ((_a4 = ownerDocument.getElementById(bag.buttonId.current)) == null ? void 0 : _a4.contains(element)) || ((_b = ownerDocument.getElementById(bag.panelId.current)) == null ? void 0 : _b.contains(element));
});
});
let closeOthers = useEvent((buttonId) => {
for (let popover of popovers) {
if (popover.buttonId.current !== buttonId)
popover.close();
}
});
let contextBag = (0, import_react36.useMemo)(
() => ({
registerPopover,
unregisterPopover,
isFocusWithinPopoverGroup,
closeOthers,
mainTreeNodeRef: root.mainTreeNodeRef
}),
[
registerPopover,
unregisterPopover,
isFocusWithinPopoverGroup,
closeOthers,
root.mainTreeNodeRef
]
);
let slot = (0, import_react36.useMemo)(() => ({}), []);
let theirProps = props;
let ourProps = { ref: groupRef };
return /* @__PURE__ */ import_react36.default.createElement(PopoverGroupContext.Provider, { value: contextBag }, render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_GROUP_TAG2,
name: "Popover.Group"
}), /* @__PURE__ */ import_react36.default.createElement(root.MainTreeNode, null));
}
var PopoverRoot = forwardRefWithAs(PopoverFn);
var Button5 = forwardRefWithAs(ButtonFn5);
var Overlay2 = forwardRefWithAs(OverlayFn2);
var Panel3 = forwardRefWithAs(PanelFn3);
var Group2 = forwardRefWithAs(GroupFn2);
var Popover = Object.assign(PopoverRoot, { Button: Button5, Overlay: Overlay2, Panel: Panel3, Group: Group2 });
// src/components/radio-group/radio-group.tsx
var import_react39 = __toESM(require("react"), 1);
// src/components/label/label.tsx
var import_react37 = __toESM(require("react"), 1);
var LabelContext = (0, import_react37.createContext)(
null
);
function useLabelContext() {
let context = (0, import_react37.useContext)(LabelContext);
if (context === null) {
let err = new Error("You used a <Label /> component, but it is not inside a relevant parent.");
if (Error.captureStackTrace)
Error.captureStackTrace(err, useLabelContext);
throw err;
}
return context;
}
function useLabels() {
let [labelIds, setLabelIds] = (0, import_react37.useState)([]);
return [
// The actual id's as string or undefined.
labelIds.length > 0 ? labelIds.join(" ") : void 0,
// The provider component
(0, import_react37.useMemo)(() => {
return function LabelProvider(props) {
let register = useEvent((value) => {
setLabelIds((existing) => [...existing, value]);
return () => setLabelIds((existing) => {
let clone = existing.slice();
let idx = clone.indexOf(value);
if (idx !== -1)
clone.splice(idx, 1);
return clone;
});
});
let contextBag = (0, import_react37.useMemo)(
() => ({ register, slot: props.slot, name: props.name, props: props.props }),
[register, props.slot, props.name, props.props]
);
return /* @__PURE__ */ import_react37.default.createElement(LabelContext.Provider, { value: contextBag }, props.children);
};
}, [setLabelIds])
];
}
var DEFAULT_LABEL_TAG3 = "label";
function LabelFn3(props, ref) {
let internalId = useId();
let { id = `headlessui-label-${internalId}`, passive = false, ...theirProps } = props;
let context = useLabelContext();
let labelRef = useSyncRefs(ref);
useIsoMorphicEffect(() => context.register(id), [id, context.register]);
let ourProps = { ref: labelRef, ...context.props, id };
if (passive) {
if ("onClick" in ourProps) {
delete ourProps["htmlFor"];
delete ourProps["onClick"];
}
if ("onClick" in theirProps) {
delete theirProps["onClick"];
}
}
return render({
ourProps,
theirProps,
slot: context.slot || {},
defaultTag: DEFAULT_LABEL_TAG3,
name: context.name || "Label"
});
}
var LabelRoot = forwardRefWithAs(LabelFn3);
var Label3 = Object.assign(LabelRoot, {
//
});
// src/hooks/use-flags.ts
var import_react38 = require("react");
function useFlags(initialFlags = 0) {
let [flags, setFlags] = (0, import_react38.useState)(initialFlags);
let mounted = useIsMounted();
let addFlag = (0, import_react38.useCallback)(
(flag) => {
if (!mounted.current)
return;
setFlags((flags2) => flags2 | flag);
},
[flags, mounted]
);
let hasFlag = (0, import_react38.useCallback)((flag) => Boolean(flags & flag), [flags]);
let removeFlag = (0, import_react38.useCallback)(
(flag) => {
if (!mounted.current)
return;
setFlags((flags2) => flags2 & ~flag);
},
[setFlags, mounted]
);
let toggleFlag = (0, import_react38.useCallback)(
(flag) => {
if (!mounted.current)
return;
setFlags((flags2) => flags2 ^ flag);
},
[setFlags]
);
return { flags, addFlag, hasFlag, removeFlag, toggleFlag };
}
// src/components/radio-group/radio-group.tsx
var reducers7 = {
[0 /* RegisterOption */](state, action) {
let nextOptions = [
...state.options,
{ id: action.id, element: action.element, propsRef: action.propsRef }
];
return {
...state,
options: sortByDomNode(nextOptions, (option) => option.element.current)
};
},
[1 /* UnregisterOption */](state, action) {
let options = state.options.slice();
let idx = state.options.findIndex((radio) => radio.id === action.id);
if (idx === -1)
return state;
options.splice(idx, 1);
return { ...state, options };
}
};
var RadioGroupDataContext = (0, import_react39.createContext)(null);
RadioGroupDataContext.displayName = "RadioGroupDataContext";
function useData3(component) {
let context = (0, import_react39.useContext)(RadioGroupDataContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <RadioGroup /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useData3);
throw err;
}
return context;
}
var RadioGroupActionsContext = (0, import_react39.createContext)(null);
RadioGroupActionsContext.displayName = "RadioGroupActionsContext";
function useActions3(component) {
let context = (0, import_react39.useContext)(RadioGroupActionsContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <RadioGroup /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useActions3);
throw err;
}
return context;
}
function stateReducer7(state, action) {
return match(action.type, reducers7, state, action);
}
var DEFAULT_RADIO_GROUP_TAG = "div";
function RadioGroupFn(props, ref) {
let internalId = useId();
let {
id = `headlessui-radiogroup-${internalId}`,
value: controlledValue,
defaultValue,
form: formName,
name,
onChange: controlledOnChange,
by = (a, z) => a === z,
disabled = false,
...theirProps
} = props;
let compare = useEvent(
typeof by === "string" ? (a, z) => {
let property = by;
return (a == null ? void 0 : a[property]) === (z == null ? void 0 : z[property]);
} : by
);
let [state, dispatch] = (0, import_react39.useReducer)(stateReducer7, { options: [] });
let options = state.options;
let [labelledby, LabelProvider] = useLabels();
let [describedby, DescriptionProvider] = useDescriptions();
let internalRadioGroupRef = (0, import_react39.useRef)(null);
let radioGroupRef = useSyncRefs(internalRadioGroupRef, ref);
let [value, onChange] = useControllable(controlledValue, controlledOnChange, defaultValue);
let firstOption = (0, import_react39.useMemo)(
() => options.find((option) => {
if (option.propsRef.current.disabled)
return false;
return true;
}),
[options]
);
let containsCheckedOption = (0, import_react39.useMemo)(
() => options.some((option) => compare(option.propsRef.current.value, value)),
[options, value]
);
let triggerChange = useEvent((nextValue) => {
var _a3;
if (disabled)
return false;
if (compare(nextValue, value))
return false;
let nextOption = (_a3 = options.find(
(option) => compare(option.propsRef.current.value, nextValue)
)) == null ? void 0 : _a3.propsRef.current;
if (nextOption == null ? void 0 : nextOption.disabled)
return false;
onChange == null ? void 0 : onChange(nextValue);
return true;
});
useTreeWalker({
container: internalRadioGroupRef.current,
accept(node) {
if (node.getAttribute("role") === "radio")
return NodeFilter.FILTER_REJECT;
if (node.hasAttribute("role"))
return NodeFilter.FILTER_SKIP;
return NodeFilter.FILTER_ACCEPT;
},
walk(node) {
node.setAttribute("role", "none");
}
});
let handleKeyDown = useEvent((event) => {
let container = internalRadioGroupRef.current;
if (!container)
return;
let ownerDocument = getOwnerDocument(container);
let all = options.filter((option) => option.propsRef.current.disabled === false).map((radio) => radio.element.current);
switch (event.key) {
case "Enter" /* Enter */:
attemptSubmit(event.currentTarget);
break;
case "ArrowLeft" /* ArrowLeft */:
case "ArrowUp" /* ArrowUp */:
{
event.preventDefault();
event.stopPropagation();
let result = focusIn(all, 2 /* Previous */ | 16 /* WrapAround */);
if (result === 2 /* Success */) {
let activeOption = options.find(
(option) => option.element.current === (ownerDocument == null ? void 0 : ownerDocument.activeElement)
);
if (activeOption)
triggerChange(activeOption.propsRef.current.value);
}
}
break;
case "ArrowRight" /* ArrowRight */:
case "ArrowDown" /* ArrowDown */:
{
event.preventDefault();
event.stopPropagation();
let result = focusIn(all, 4 /* Next */ | 16 /* WrapAround */);
if (result === 2 /* Success */) {
let activeOption = options.find(
(option) => option.element.current === (ownerDocument == null ? void 0 : ownerDocument.activeElement)
);
if (activeOption)
triggerChange(activeOption.propsRef.current.value);
}
}
break;
case " " /* Space */:
{
event.preventDefault();
event.stopPropagation();
let activeOption = options.find(
(option) => option.element.current === (ownerDocument == null ? void 0 : ownerDocument.activeElement)
);
if (activeOption)
triggerChange(activeOption.propsRef.current.value);
}
break;
}
});
let registerOption = useEvent((option) => {
dispatch({ type: 0 /* RegisterOption */, ...option });
return () => dispatch({ type: 1 /* UnregisterOption */, id: option.id });
});
let radioGroupData = (0, import_react39.useMemo)(
() => ({ value, firstOption, containsCheckedOption, disabled, compare, ...state }),
[value, firstOption, containsCheckedOption, disabled, compare, state]
);
let radioGroupActions = (0, import_react39.useMemo)(
() => ({ registerOption, change: triggerChange }),
[registerOption, triggerChange]
);
let ourProps = {
ref: radioGroupRef,
id,
role: "radiogroup",
"aria-labelledby": labelledby,
"aria-describedby": describedby,
onKeyDown: handleKeyDown
};
let slot = (0, import_react39.useMemo)(() => ({ value }), [value]);
let form = (0, import_react39.useRef)(null);
let d = useDisposables();
(0, import_react39.useEffect)(() => {
if (!form.current)
return;
if (defaultValue === void 0)
return;
d.addEventListener(form.current, "reset", () => {
triggerChange(defaultValue);
});
}, [
form,
triggerChange
/* Explicitly ignoring `defaultValue` */
]);
return /* @__PURE__ */ import_react39.default.createElement(DescriptionProvider, { name: "RadioGroup.Description" }, /* @__PURE__ */ import_react39.default.createElement(LabelProvider, { name: "RadioGroup.Label" }, /* @__PURE__ */ import_react39.default.createElement(RadioGroupActionsContext.Provider, { value: radioGroupActions }, /* @__PURE__ */ import_react39.default.createElement(RadioGroupDataContext.Provider, { value: radioGroupData }, name != null && value != null && objectToFormEntries({ [name]: value }).map(([name2, value2], idx) => /* @__PURE__ */ import_react39.default.createElement(
Hidden,
{
features: 4 /* Hidden */,
ref: idx === 0 ? (element) => {
var _a3;
form.current = (_a3 = element == null ? void 0 : element.closest("form")) != null ? _a3 : null;
} : void 0,
...compact({
key: name2,
as: "input",
type: "radio",
checked: value2 != null,
hidden: true,
readOnly: true,
form: formName,
disabled,
name: name2,
value: value2
})
}
)), render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_RADIO_GROUP_TAG,
name: "RadioGroup"
})))));
}
var DEFAULT_OPTION_TAG3 = "div";
function OptionFn3(props, ref) {
var _a3;
let internalId = useId();
let {
id = `headlessui-radiogroup-option-${internalId}`,
value,
disabled = false,
...theirProps
} = props;
let internalOptionRef = (0, import_react39.useRef)(null);
let optionRef = useSyncRefs(internalOptionRef, ref);
let [labelledby, LabelProvider] = useLabels();
let [describedby, DescriptionProvider] = useDescriptions();
let { addFlag, removeFlag, hasFlag } = useFlags(1 /* Empty */);
let propsRef = useLatestValue({ value, disabled });
let data = useData3("RadioGroup.Option");
let actions = useActions3("RadioGroup.Option");
useIsoMorphicEffect(
() => actions.registerOption({ id, element: internalOptionRef, propsRef }),
[id, actions, internalOptionRef, propsRef]
);
let handleClick = useEvent((event) => {
var _a4;
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
if (!actions.change(value))
return;
addFlag(2 /* Active */);
(_a4 = internalOptionRef.current) == null ? void 0 : _a4.focus();
});
let handleFocus = useEvent((event) => {
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
addFlag(2 /* Active */);
});
let handleBlur = useEvent(() => removeFlag(2 /* Active */));
let isFirstOption = ((_a3 = data.firstOption) == null ? void 0 : _a3.id) === id;
let isDisabled = data.disabled || disabled;
let checked = data.compare(data.value, value);
let ourProps = {
ref: optionRef,
id,
role: "radio",
"aria-checked": checked ? "true" : "false",
"aria-labelledby": labelledby,
"aria-describedby": describedby,
"aria-disabled": isDisabled ? true : void 0,
tabIndex: (() => {
if (isDisabled)
return -1;
if (checked)
return 0;
if (!data.containsCheckedOption && isFirstOption)
return 0;
return -1;
})(),
onClick: isDisabled ? void 0 : handleClick,
onFocus: isDisabled ? void 0 : handleFocus,
onBlur: isDisabled ? void 0 : handleBlur
};
let slot = (0, import_react39.useMemo)(
() => ({ checked, disabled: isDisabled, active: hasFlag(2 /* Active */) }),
[checked, isDisabled, hasFlag]
);
return /* @__PURE__ */ import_react39.default.createElement(DescriptionProvider, { name: "RadioGroup.Description" }, /* @__PURE__ */ import_react39.default.createElement(LabelProvider, { name: "RadioGroup.Label" }, render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_OPTION_TAG3,
name: "RadioGroup.Option"
})));
}
var RadioGroupRoot = forwardRefWithAs(RadioGroupFn);
var Option3 = forwardRefWithAs(OptionFn3);
var RadioGroup = Object.assign(RadioGroupRoot, {
Option: Option3,
Label: Label3,
Description
});
// src/components/switch/switch.tsx
var import_react40 = __toESM(require("react"), 1);
var GroupContext = (0, import_react40.createContext)(null);
GroupContext.displayName = "GroupContext";
var DEFAULT_GROUP_TAG3 = import_react40.Fragment;
function GroupFn3(props) {
var _a3;
let [switchElement, setSwitchElement] = (0, import_react40.useState)(null);
let [labelledby, LabelProvider] = useLabels();
let [describedby, DescriptionProvider] = useDescriptions();
let context = (0, import_react40.useMemo)(
() => ({ switch: switchElement, setSwitch: setSwitchElement, labelledby, describedby }),
[switchElement, setSwitchElement, labelledby, describedby]
);
let ourProps = {};
let theirProps = props;
return /* @__PURE__ */ import_react40.default.createElement(DescriptionProvider, { name: "Switch.Description" }, /* @__PURE__ */ import_react40.default.createElement(
LabelProvider,
{
name: "Switch.Label",
props: {
htmlFor: (_a3 = context.switch) == null ? void 0 : _a3.id,
onClick(event) {
if (!switchElement)
return;
if (event.currentTarget.tagName === "LABEL") {
event.preventDefault();
}
switchElement.click();
switchElement.focus({ preventScroll: true });
}
}
},
/* @__PURE__ */ import_react40.default.createElement(GroupContext.Provider, { value: context }, render({
ourProps,
theirProps,
defaultTag: DEFAULT_GROUP_TAG3,
name: "Switch.Group"
}))
));
}
var DEFAULT_SWITCH_TAG = "button";
function SwitchFn(props, ref) {
var _a3;
let internalId = useId();
let {
id = `headlessui-switch-${internalId}`,
checked: controlledChecked,
defaultChecked = false,
onChange: controlledOnChange,
disabled = false,
name,
value,
form,
...theirProps
} = props;
let groupContext = (0, import_react40.useContext)(GroupContext);
let internalSwitchRef = (0, import_react40.useRef)(null);
let switchRef = useSyncRefs(
internalSwitchRef,
ref,
groupContext === null ? null : groupContext.setSwitch
);
let [checked, onChange] = useControllable(controlledChecked, controlledOnChange, defaultChecked);
let toggle = useEvent(() => onChange == null ? void 0 : onChange(!checked));
let handleClick = useEvent((event) => {
if (isDisabledReactIssue7711(event.currentTarget))
return event.preventDefault();
event.preventDefault();
toggle();
});
let handleKeyUp = useEvent((event) => {
if (event.key === " " /* Space */) {
event.preventDefault();
toggle();
} else if (event.key === "Enter" /* Enter */) {
attemptSubmit(event.currentTarget);
}
});
let handleKeyPress = useEvent((event) => event.preventDefault());
let slot = (0, import_react40.useMemo)(() => ({ checked }), [checked]);
let ourProps = {
id,
ref: switchRef,
role: "switch",
type: useResolveButtonType(props, internalSwitchRef),
tabIndex: props.tabIndex === -1 ? 0 : (_a3 = props.tabIndex) != null ? _a3 : 0,
"aria-checked": checked,
"aria-labelledby": groupContext == null ? void 0 : groupContext.labelledby,
"aria-describedby": groupContext == null ? void 0 : groupContext.describedby,
disabled,
onClick: handleClick,
onKeyUp: handleKeyUp,
onKeyPress: handleKeyPress
};
let d = useDisposables();
(0, import_react40.useEffect)(() => {
var _a4;
let form2 = (_a4 = internalSwitchRef.current) == null ? void 0 : _a4.closest("form");
if (!form2)
return;
if (defaultChecked === void 0)
return;
d.addEventListener(form2, "reset", () => {
onChange(defaultChecked);
});
}, [
internalSwitchRef,
onChange
/* Explicitly ignoring `defaultValue` */
]);
return /* @__PURE__ */ import_react40.default.createElement(import_react40.default.Fragment, null, name != null && checked && /* @__PURE__ */ import_react40.default.createElement(
Hidden,
{
features: 4 /* Hidden */,
...compact({
as: "input",
type: "checkbox",
hidden: true,
readOnly: true,
disabled,
form,
checked,
name,
value
})
}
), render({ ourProps, theirProps, slot, defaultTag: DEFAULT_SWITCH_TAG, name: "Switch" }));
}
var SwitchRoot = forwardRefWithAs(SwitchFn);
var Group3 = GroupFn3;
var Switch = Object.assign(SwitchRoot, {
Group: Group3,
Label: Label3,
Description
});
// src/components/tabs/tabs.tsx
var import_react42 = __toESM(require("react"), 1);
// src/internal/focus-sentinel.tsx
var import_react41 = __toESM(require("react"), 1);
function FocusSentinel({ onFocus }) {
let [enabled, setEnabled] = (0, import_react41.useState)(true);
let mounted = useIsMounted();
if (!enabled)
return null;
return /* @__PURE__ */ import_react41.default.createElement(
Hidden,
{
as: "button",
type: "button",
features: 2 /* Focusable */,
onFocus: (event) => {
event.preventDefault();
let frame;
let tries = 50;
function forwardFocus() {
if (tries-- <= 0) {
if (frame)
cancelAnimationFrame(frame);
return;
}
if (onFocus()) {
cancelAnimationFrame(frame);
if (!mounted.current)
return;
setEnabled(false);
return;
}
frame = requestAnimationFrame(forwardFocus);
}
frame = requestAnimationFrame(forwardFocus);
}
}
);
}
// src/utils/stable-collection.tsx
var React25 = __toESM(require("react"), 1);
var StableCollectionContext = React25.createContext(null);
function createCollection() {
return {
/** @type {Map<string, Map<string, number>>} */
groups: /* @__PURE__ */ new Map(),
get(group, key) {
var _a3;
let list = this.groups.get(group);
if (!list) {
list = /* @__PURE__ */ new Map();
this.groups.set(group, list);
}
let renders = (_a3 = list.get(key)) != null ? _a3 : 0;
list.set(key, renders + 1);
let index = Array.from(list.keys()).indexOf(key);
function release() {
let renders2 = list.get(key);
if (renders2 > 1) {
list.set(key, renders2 - 1);
} else {
list.delete(key);
}
}
return [index, release];
}
};
}
function StableCollection({ children }) {
let collection = React25.useRef(createCollection());
return /* @__PURE__ */ React25.createElement(StableCollectionContext.Provider, { value: collection }, children);
}
function useStableCollectionIndex(group) {
let collection = React25.useContext(StableCollectionContext);
if (!collection)
throw new Error("You must wrap your component in a <StableCollection>");
let key = useStableCollectionKey();
let [idx, cleanupIdx] = collection.current.get(group, key);
React25.useEffect(() => cleanupIdx, []);
return idx;
}
function useStableCollectionKey() {
var _a3, _b, _c;
let owner = (
// @ts-ignore
(_c = (_b = (_a3 = React25.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED) == null ? void 0 : _a3.ReactCurrentOwner) == null ? void 0 : _b.current) != null ? _c : null
);
if (!owner)
return Symbol();
let indexes = [];
let fiber = owner;
while (fiber) {
indexes.push(fiber.index);
fiber = fiber.return;
}
return "$." + indexes.join(".");
}
// src/components/tabs/tabs.tsx
var reducers8 = {
[0 /* SetSelectedIndex */](state, action) {
var _a3;
let tabs = sortByDomNode(state.tabs, (tab) => tab.current);
let panels = sortByDomNode(state.panels, (panel) => panel.current);
let focusableTabs = tabs.filter((tab) => {
var _a4;
return !((_a4 = tab.current) == null ? void 0 : _a4.hasAttribute("disabled"));
});
let nextState = { ...state, tabs, panels };
if (
// Underflow
action.index < 0 || // Overflow
action.index > tabs.length - 1
) {
let direction = match(Math.sign(action.index - state.selectedIndex), {
[-1 /* Less */]: () => 1 /* Backwards */,
[0 /* Equal */]: () => {
return match(Math.sign(action.index), {
[-1 /* Less */]: () => 0 /* Forwards */,
[0 /* Equal */]: () => 0 /* Forwards */,
[1 /* Greater */]: () => 1 /* Backwards */
});
},
[1 /* Greater */]: () => 0 /* Forwards */
});
if (focusableTabs.length === 0) {
return nextState;
}
let nextSelectedIndex = match(direction, {
[0 /* Forwards */]: () => tabs.indexOf(focusableTabs[0]),
[1 /* Backwards */]: () => tabs.indexOf(focusableTabs[focusableTabs.length - 1])
});
return {
...nextState,
selectedIndex: nextSelectedIndex === -1 ? state.selectedIndex : nextSelectedIndex
};
}
let before = tabs.slice(0, action.index);
let after = tabs.slice(action.index);
let next = [...after, ...before].find((tab) => focusableTabs.includes(tab));
if (!next)
return nextState;
let selectedIndex = (_a3 = tabs.indexOf(next)) != null ? _a3 : state.selectedIndex;
if (selectedIndex === -1)
selectedIndex = state.selectedIndex;
return { ...nextState, selectedIndex };
},
[1 /* RegisterTab */](state, action) {
if (state.tabs.includes(action.tab))
return state;
let activeTab = state.tabs[state.selectedIndex];
let adjustedTabs = sortByDomNode([...state.tabs, action.tab], (tab) => tab.current);
let selectedIndex = state.selectedIndex;
if (!state.info.current.isControlled) {
selectedIndex = adjustedTabs.indexOf(activeTab);
if (selectedIndex === -1)
selectedIndex = state.selectedIndex;
}
return { ...state, tabs: adjustedTabs, selectedIndex };
},
[2 /* UnregisterTab */](state, action) {
return { ...state, tabs: state.tabs.filter((tab) => tab !== action.tab) };
},
[3 /* RegisterPanel */](state, action) {
if (state.panels.includes(action.panel))
return state;
return {
...state,
panels: sortByDomNode([...state.panels, action.panel], (panel) => panel.current)
};
},
[4 /* UnregisterPanel */](state, action) {
return { ...state, panels: state.panels.filter((panel) => panel !== action.panel) };
}
};
var TabsDataContext = (0, import_react42.createContext)(null);
TabsDataContext.displayName = "TabsDataContext";
function useData4(component) {
let context = (0, import_react42.useContext)(TabsDataContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Tab.Group /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useData4);
throw err;
}
return context;
}
var TabsActionsContext = (0, import_react42.createContext)(null);
TabsActionsContext.displayName = "TabsActionsContext";
function useActions4(component) {
let context = (0, import_react42.useContext)(TabsActionsContext);
if (context === null) {
let err = new Error(`<${component} /> is missing a parent <Tab.Group /> component.`);
if (Error.captureStackTrace)
Error.captureStackTrace(err, useActions4);
throw err;
}
return context;
}
function stateReducer8(state, action) {
return match(action.type, reducers8, state, action);
}
var DEFAULT_TABS_TAG = import_react42.Fragment;
function GroupFn4(props, ref) {
let {
defaultIndex = 0,
vertical = false,
manual = false,
onChange,
selectedIndex = null,
...theirProps
} = props;
const orientation = vertical ? "vertical" : "horizontal";
const activation = manual ? "manual" : "auto";
let isControlled = selectedIndex !== null;
let info = useLatestValue({ isControlled });
let tabsRef = useSyncRefs(ref);
let [state, dispatch] = (0, import_react42.useReducer)(stateReducer8, {
info,
selectedIndex: selectedIndex != null ? selectedIndex : defaultIndex,
tabs: [],
panels: []
});
let slot = (0, import_react42.useMemo)(() => ({ selectedIndex: state.selectedIndex }), [state.selectedIndex]);
let onChangeRef = useLatestValue(onChange || (() => {
}));
let stableTabsRef = useLatestValue(state.tabs);
let tabsData = (0, import_react42.useMemo)(
() => ({ orientation, activation, ...state }),
[orientation, activation, state]
);
let registerTab = useEvent((tab) => {
dispatch({ type: 1 /* RegisterTab */, tab });
return () => dispatch({ type: 2 /* UnregisterTab */, tab });
});
let registerPanel = useEvent((panel) => {
dispatch({ type: 3 /* RegisterPanel */, panel });
return () => dispatch({ type: 4 /* UnregisterPanel */, panel });
});
let change = useEvent((index) => {
if (realSelectedIndex.current !== index) {
onChangeRef.current(index);
}
if (!isControlled) {
dispatch({ type: 0 /* SetSelectedIndex */, index });
}
});
let realSelectedIndex = useLatestValue(isControlled ? props.selectedIndex : state.selectedIndex);
let tabsActions = (0, import_react42.useMemo)(() => ({ registerTab, registerPanel, change }), []);
useIsoMorphicEffect(() => {
dispatch({ type: 0 /* SetSelectedIndex */, index: selectedIndex != null ? selectedIndex : defaultIndex });
}, [
selectedIndex
/* Deliberately skipping defaultIndex */
]);
useIsoMorphicEffect(() => {
if (realSelectedIndex.current === void 0)
return;
if (state.tabs.length <= 0)
return;
let sorted = sortByDomNode(state.tabs, (tab) => tab.current);
let didOrderChange = sorted.some((tab, i) => state.tabs[i] !== tab);
if (didOrderChange) {
change(sorted.indexOf(state.tabs[realSelectedIndex.current]));
}
});
let ourProps = { ref: tabsRef };
return /* @__PURE__ */ import_react42.default.createElement(StableCollection, null, /* @__PURE__ */ import_react42.default.createElement(TabsActionsContext.Provider, { value: tabsActions }, /* @__PURE__ */ import_react42.default.createElement(TabsDataContext.Provider, { value: tabsData }, tabsData.tabs.length <= 0 && /* @__PURE__ */ import_react42.default.createElement(
FocusSentinel,
{
onFocus: () => {
var _a3, _b;
for (let tab of stableTabsRef.current) {
if (((_a3 = tab.current) == null ? void 0 : _a3.tabIndex) === 0) {
(_b = tab.current) == null ? void 0 : _b.focus();
return true;
}
}
return false;
}
}
), render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_TABS_TAG,
name: "Tabs"
}))));
}
var DEFAULT_LIST_TAG = "div";
function ListFn(props, ref) {
let { orientation, selectedIndex } = useData4("Tab.List");
let listRef = useSyncRefs(ref);
let slot = { selectedIndex };
let theirProps = props;
let ourProps = {
ref: listRef,
role: "tablist",
"aria-orientation": orientation
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_LIST_TAG,
name: "Tabs.List"
});
}
var DEFAULT_TAB_TAG = "button";
function TabFn(props, ref) {
var _a3, _b;
let internalId = useId();
let { id = `headlessui-tabs-tab-${internalId}`, ...theirProps } = props;
let { orientation, activation, selectedIndex, tabs, panels } = useData4("Tab");
let actions = useActions4("Tab");
let data = useData4("Tab");
let internalTabRef = (0, import_react42.useRef)(null);
let tabRef = useSyncRefs(internalTabRef, ref);
useIsoMorphicEffect(() => actions.registerTab(internalTabRef), [actions, internalTabRef]);
let mySSRIndex = useStableCollectionIndex("tabs");
let myIndex = tabs.indexOf(internalTabRef);
if (myIndex === -1)
myIndex = mySSRIndex;
let selected = myIndex === selectedIndex;
let activateUsing = useEvent((cb) => {
var _a4;
let result = cb();
if (result === 2 /* Success */ && activation === "auto") {
let newTab = (_a4 = getOwnerDocument(internalTabRef)) == null ? void 0 : _a4.activeElement;
let idx = data.tabs.findIndex((tab) => tab.current === newTab);
if (idx !== -1)
actions.change(idx);
}
return result;
});
let handleKeyDown = useEvent((event) => {
let list = tabs.map((tab) => tab.current).filter(Boolean);
if (event.key === " " /* Space */ || event.key === "Enter" /* Enter */) {
event.preventDefault();
event.stopPropagation();
actions.change(myIndex);
return;
}
switch (event.key) {
case "Home" /* Home */:
case "PageUp" /* PageUp */:
event.preventDefault();
event.stopPropagation();
return activateUsing(() => focusIn(list, 1 /* First */));
case "End" /* End */:
case "PageDown" /* PageDown */:
event.preventDefault();
event.stopPropagation();
return activateUsing(() => focusIn(list, 8 /* Last */));
}
let result = activateUsing(() => {
return match(orientation, {
vertical() {
if (event.key === "ArrowUp" /* ArrowUp */)
return focusIn(list, 2 /* Previous */ | 16 /* WrapAround */);
if (event.key === "ArrowDown" /* ArrowDown */)
return focusIn(list, 4 /* Next */ | 16 /* WrapAround */);
return 0 /* Error */;
},
horizontal() {
if (event.key === "ArrowLeft" /* ArrowLeft */)
return focusIn(list, 2 /* Previous */ | 16 /* WrapAround */);
if (event.key === "ArrowRight" /* ArrowRight */)
return focusIn(list, 4 /* Next */ | 16 /* WrapAround */);
return 0 /* Error */;
}
});
});
if (result === 2 /* Success */) {
return event.preventDefault();
}
});
let ready = (0, import_react42.useRef)(false);
let handleSelection = useEvent(() => {
var _a4;
if (ready.current)
return;
ready.current = true;
(_a4 = internalTabRef.current) == null ? void 0 : _a4.focus({ preventScroll: true });
actions.change(myIndex);
microTask(() => {
ready.current = false;
});
});
let handleMouseDown = useEvent((event) => {
event.preventDefault();
});
let slot = (0, import_react42.useMemo)(
() => {
var _a4;
return { selected, disabled: (_a4 = props.disabled) != null ? _a4 : false };
},
[selected, props.disabled]
);
let ourProps = {
ref: tabRef,
onKeyDown: handleKeyDown,
onMouseDown: handleMouseDown,
onClick: handleSelection,
id,
role: "tab",
type: useResolveButtonType(props, internalTabRef),
"aria-controls": (_b = (_a3 = panels[myIndex]) == null ? void 0 : _a3.current) == null ? void 0 : _b.id,
"aria-selected": selected,
tabIndex: selected ? 0 : -1
};
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_TAB_TAG,
name: "Tabs.Tab"
});
}
var DEFAULT_PANELS_TAG = "div";
function PanelsFn(props, ref) {
let { selectedIndex } = useData4("Tab.Panels");
let panelsRef = useSyncRefs(ref);
let slot = (0, import_react42.useMemo)(() => ({ selectedIndex }), [selectedIndex]);
let theirProps = props;
let ourProps = { ref: panelsRef };
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_PANELS_TAG,
name: "Tabs.Panels"
});
}
var DEFAULT_PANEL_TAG4 = "div";
var PanelRenderFeatures3 = 1 /* RenderStrategy */ | 2 /* Static */;
function PanelFn4(props, ref) {
var _a3, _b, _c, _d;
let internalId = useId();
let { id = `headlessui-tabs-panel-${internalId}`, tabIndex = 0, ...theirProps } = props;
let { selectedIndex, tabs, panels } = useData4("Tab.Panel");
let actions = useActions4("Tab.Panel");
let internalPanelRef = (0, import_react42.useRef)(null);
let panelRef = useSyncRefs(internalPanelRef, ref);
useIsoMorphicEffect(
() => actions.registerPanel(internalPanelRef),
[
actions,
internalPanelRef,
// The `id` prop is here to force a re-render of the
// corresponding `Tab` whenever the `id` is calculated in React < 18
id
]
);
let mySSRIndex = useStableCollectionIndex("panels");
let myIndex = panels.indexOf(internalPanelRef);
if (myIndex === -1)
myIndex = mySSRIndex;
let selected = myIndex === selectedIndex;
let slot = (0, import_react42.useMemo)(() => ({ selected }), [selected]);
let ourProps = {
ref: panelRef,
id,
role: "tabpanel",
"aria-labelledby": (_b = (_a3 = tabs[myIndex]) == null ? void 0 : _a3.current) == null ? void 0 : _b.id,
tabIndex: selected ? tabIndex : -1
};
if (!selected && ((_c = theirProps.unmount) != null ? _c : true) && !((_d = theirProps.static) != null ? _d : false)) {
return /* @__PURE__ */ import_react42.default.createElement(Hidden, { as: "span", "aria-hidden": "true", ...ourProps });
}
return render({
ourProps,
theirProps,
slot,
defaultTag: DEFAULT_PANEL_TAG4,
features: PanelRenderFeatures3,
visible: selected,
name: "Tabs.Panel"
});
}
var TabRoot = forwardRefWithAs(TabFn);
var Group4 = forwardRefWithAs(GroupFn4);
var List = forwardRefWithAs(ListFn);
var Panels = forwardRefWithAs(PanelsFn);
var Panel4 = forwardRefWithAs(PanelFn4);
var Tab = Object.assign(TabRoot, { Group: Group4, List, Panels, Panel: Panel4 });
// src/components/transitions/transition.tsx
var import_react43 = __toESM(require("react"), 1);
// src/utils/once.ts
function once(cb) {
let state = { called: false };
return (...args) => {
if (state.called)
return;
state.called = true;
return cb(...args);
};
}
// src/components/transitions/utils/transition.ts
function addClasses(node, ...classes) {
node && classes.length > 0 && node.classList.add(...classes);
}
function removeClasses(node, ...classes) {
node && classes.length > 0 && node.classList.remove(...classes);
}
function waitForTransition(node, done) {
let d = disposables();
if (!node)
return d.dispose;
let { transitionDuration, transitionDelay } = getComputedStyle(node);
let [durationMs, delayMs] = [transitionDuration, transitionDelay].map((value) => {
let [resolvedValue = 0] = value.split(",").filter(Boolean).map((v) => v.includes("ms") ? parseFloat(v) : parseFloat(v) * 1e3).sort((a, z) => z - a);
return resolvedValue;
});
let totalDuration = durationMs + delayMs;
if (totalDuration !== 0) {
if (false) {
let dispose = d.setTimeout(() => {
done();
dispose();
}, totalDuration);
} else {
d.group((d2) => {
d2.setTimeout(() => {
done();
d2.dispose();
}, totalDuration);
d2.addEventListener(node, "transitionrun", (event) => {
if (event.target !== event.currentTarget)
return;
d2.dispose();
});
});
let dispose = d.addEventListener(node, "transitionend", (event) => {
if (event.target !== event.currentTarget)
return;
done();
dispose();
});
}
} else {
done();
}
d.add(() => done());
return d.dispose;
}
function transition(node, classes, show, done) {
let direction = show ? "enter" : "leave";
let d = disposables();
let _done = done !== void 0 ? once(done) : () => {
};
if (direction === "enter") {
node.removeAttribute("hidden");
node.style.display = "";
}
let base = match(direction, {
enter: () => classes.enter,
leave: () => classes.leave
});
let to = match(direction, {
enter: () => classes.enterTo,
leave: () => classes.leaveTo
});
let from = match(direction, {
enter: () => classes.enterFrom,
leave: () => classes.leaveFrom
});
removeClasses(
node,
...classes.base,
...classes.enter,
...classes.enterTo,
...classes.enterFrom,
...classes.leave,
...classes.leaveFrom,
...classes.leaveTo,
...classes.entered
);
addClasses(node, ...classes.base, ...base, ...from);
d.nextFrame(() => {
removeClasses(node, ...classes.base, ...base, ...from);
addClasses(node, ...classes.base, ...base, ...to);
waitForTransition(node, () => {
removeClasses(node, ...classes.base, ...base);
addClasses(node, ...classes.base, ...classes.entered);
return _done();
});
});
return d.dispose;
}
// src/hooks/use-transition.ts
function useTransition({
immediate,
container,
direction,
classes,
onStart,
onStop
}) {
let mounted = useIsMounted();
let d = useDisposables();
let latestDirection = useLatestValue(direction);
useIsoMorphicEffect(() => {
if (!immediate)
return;
latestDirection.current = "enter";
}, [immediate]);
useIsoMorphicEffect(() => {
let dd = disposables();
d.add(dd.dispose);
let node = container.current;
if (!node)
return;
if (latestDirection.current === "idle")
return;
if (!mounted.current)
return;
dd.dispose();
onStart.current(latestDirection.current);
dd.add(
transition(node, classes.current, latestDirection.current === "enter", () => {
dd.dispose();
onStop.current(latestDirection.current);
})
);
return dd.dispose;
}, [direction]);
}
// src/components/transitions/transition.tsx
function splitClasses(classes = "") {
return classes.split(/\s+/).filter((className) => className.length > 1);
}
var TransitionContext = (0, import_react43.createContext)(null);
TransitionContext.displayName = "TransitionContext";
function useTransitionContext() {
let context = (0, import_react43.useContext)(TransitionContext);
if (context === null) {
throw new Error(
"A <Transition.Child /> is used but it is missing a parent <Transition /> or <Transition.Root />."
);
}
return context;
}
function useParentNesting() {
let context = (0, import_react43.useContext)(NestingContext);
if (context === null) {
throw new Error(
"A <Transition.Child /> is used but it is missing a parent <Transition /> or <Transition.Root />."
);
}
return context;
}
var NestingContext = (0, import_react43.createContext)(null);
NestingContext.displayName = "NestingContext";
function hasChildren(bag) {
if ("children" in bag)
return hasChildren(bag.children);
return bag.current.filter(({ el }) => el.current !== null).filter(({ state }) => state === "visible" /* Visible */).length > 0;
}
function useNesting(done, parent) {
let doneRef = useLatestValue(done);
let transitionableChildren = (0, import_react43.useRef)([]);
let mounted = useIsMounted();
let d = useDisposables();
let unregister = useEvent((container, strategy = 1 /* Hidden */) => {
let idx = transitionableChildren.current.findIndex(({ el }) => el === container);
if (idx === -1)
return;
match(strategy, {
[0 /* Unmount */]() {
transitionableChildren.current.splice(idx, 1);
},
[1 /* Hidden */]() {
transitionableChildren.current[idx].state = "hidden" /* Hidden */;
}
});
d.microTask(() => {
var _a3;
if (!hasChildren(transitionableChildren) && mounted.current) {
(_a3 = doneRef.current) == null ? void 0 : _a3.call(doneRef);
}
});
});
let register = useEvent((container) => {
let child = transitionableChildren.current.find(({ el }) => el === container);
if (!child) {
transitionableChildren.current.push({ el: container, state: "visible" /* Visible */ });
} else if (child.state !== "visible" /* Visible */) {
child.state = "visible" /* Visible */;
}
return () => unregister(container, 0 /* Unmount */);
});
let todos = (0, import_react43.useRef)([]);
let wait = (0, import_react43.useRef)(Promise.resolve());
let chains = (0, import_react43.useRef)({
enter: [],
leave: [],
idle: []
});
let onStart = useEvent(
(container, direction, cb) => {
todos.current.splice(0);
if (parent) {
parent.chains.current[direction] = parent.chains.current[direction].filter(
([containerInParent]) => containerInParent !== container
);
}
parent == null ? void 0 : parent.chains.current[direction].push([
container,
new Promise((resolve) => {
todos.current.push(resolve);
})
]);
parent == null ? void 0 : parent.chains.current[direction].push([
container,
new Promise((resolve) => {
Promise.all(chains.current[direction].map(([_container, promise]) => promise)).then(
() => resolve()
);
})
]);
if (direction === "enter") {
wait.current = wait.current.then(() => parent == null ? void 0 : parent.wait.current).then(() => cb(direction));
} else {
cb(direction);
}
}
);
let onStop = useEvent(
(_container, direction, cb) => {
Promise.all(chains.current[direction].splice(0).map(([_container2, promise]) => promise)).then(() => {
var _a3;
(_a3 = todos.current.shift()) == null ? void 0 : _a3();
}).then(() => cb(direction));
}
);
return (0, import_react43.useMemo)(
() => ({
children: transitionableChildren,
register,
unregister,
onStart,
onStop,
wait,
chains
}),
[register, unregister, transitionableChildren, onStart, onStop, chains, wait]
);
}
function noop() {
}
var eventNames = ["beforeEnter", "afterEnter", "beforeLeave", "afterLeave"];
function ensureEventHooksExist(events) {
var _a3;
let result = {};
for (let name of eventNames) {
result[name] = (_a3 = events[name]) != null ? _a3 : noop;
}
return result;
}
function useEvents(events) {
let eventsRef = (0, import_react43.useRef)(ensureEventHooksExist(events));
(0, import_react43.useEffect)(() => {
eventsRef.current = ensureEventHooksExist(events);
}, [events]);
return eventsRef;
}
var DEFAULT_TRANSITION_CHILD_TAG = "div";
var TransitionChildRenderFeatures = 1 /* RenderStrategy */;
function TransitionChildFn(props, ref) {
var _a3, _b;
let {
// Event "handlers"
beforeEnter,
afterEnter,
beforeLeave,
afterLeave,
// Class names
enter,
enterFrom,
enterTo,
entered,
leave,
leaveFrom,
leaveTo,
// @ts-expect-error
...rest
} = props;
let container = (0, import_react43.useRef)(null);
let transitionRef = useSyncRefs(container, ref);
let strategy = ((_a3 = rest.unmount) != null ? _a3 : true) ? 0 /* Unmount */ : 1 /* Hidden */;
let { show, appear, initial } = useTransitionContext();
let [state, setState] = (0, import_react43.useState)(show ? "visible" /* Visible */ : "hidden" /* Hidden */);
let parentNesting = useParentNesting();
let { register, unregister } = parentNesting;
(0, import_react43.useEffect)(() => register(container), [register, container]);
(0, import_react43.useEffect)(() => {
if (strategy !== 1 /* Hidden */)
return;
if (!container.current)
return;
if (show && state !== "visible" /* Visible */) {
setState("visible" /* Visible */);
return;
}
return match(state, {
["hidden" /* Hidden */]: () => unregister(container),
["visible" /* Visible */]: () => register(container)
});
}, [state, container, register, unregister, show, strategy]);
let classes = useLatestValue({
base: splitClasses(rest.className),
enter: splitClasses(enter),
enterFrom: splitClasses(enterFrom),
enterTo: splitClasses(enterTo),
entered: splitClasses(entered),
leave: splitClasses(leave),
leaveFrom: splitClasses(leaveFrom),
leaveTo: splitClasses(leaveTo)
});
let events = useEvents({
beforeEnter,
afterEnter,
beforeLeave,
afterLeave
});
let ready = useServerHandoffComplete();
(0, import_react43.useEffect)(() => {
if (ready && state === "visible" /* Visible */ && container.current === null) {
throw new Error("Did you forget to passthrough the `ref` to the actual DOM node?");
}
}, [container, state, ready]);
let skip = initial && !appear;
let immediate = appear && show && initial;
let transitionDirection = (() => {
if (!ready)
return "idle";
if (skip)
return "idle";
return show ? "enter" : "leave";
})();
let transitionStateFlags = useFlags(0);
let beforeEvent = useEvent((direction) => {
return match(direction, {
enter: () => {
transitionStateFlags.addFlag(8 /* Opening */);
events.current.beforeEnter();
},
leave: () => {
transitionStateFlags.addFlag(4 /* Closing */);
events.current.beforeLeave();
},
idle: () => {
}
});
});
let afterEvent = useEvent((direction) => {
return match(direction, {
enter: () => {
transitionStateFlags.removeFlag(8 /* Opening */);
events.current.afterEnter();
},
leave: () => {
transitionStateFlags.removeFlag(4 /* Closing */);
events.current.afterLeave();
},
idle: () => {
}
});
});
let nesting = useNesting(() => {
setState("hidden" /* Hidden */);
unregister(container);
}, parentNesting);
let isTransitioning = (0, import_react43.useRef)(false);
useTransition({
immediate,
container,
classes,
direction: transitionDirection,
onStart: useLatestValue((direction) => {
isTransitioning.current = true;
nesting.onStart(container, direction, beforeEvent);
}),
onStop: useLatestValue((direction) => {
isTransitioning.current = false;
nesting.onStop(container, direction, afterEvent);
if (direction === "leave" && !hasChildren(nesting)) {
setState("hidden" /* Hidden */);
unregister(container);
}
})
});
let theirProps = rest;
let ourProps = { ref: transitionRef };
if (immediate) {
theirProps = {
...theirProps,
// Already apply the `enter` and `enterFrom` on the server if required
className: classNames(rest.className, ...classes.current.enter, ...classes.current.enterFrom)
};
} else if (isTransitioning.current) {
theirProps.className = classNames(rest.className, (_b = container.current) == null ? void 0 : _b.className);
if (theirProps.className === "")
delete theirProps.className;
}
return /* @__PURE__ */ import_react43.default.createElement(NestingContext.Provider, { value: nesting }, /* @__PURE__ */ import_react43.default.createElement(
OpenClosedProvider,
{
value: match(state, {
["visible" /* Visible */]: 1 /* Open */,
["hidden" /* Hidden */]: 2 /* Closed */
}) | transitionStateFlags.flags
},
render({
ourProps,
theirProps,
defaultTag: DEFAULT_TRANSITION_CHILD_TAG,
features: TransitionChildRenderFeatures,
visible: state === "visible" /* Visible */,
name: "Transition.Child"
})
));
}
function TransitionRootFn(props, ref) {
let { show, appear = false, unmount = true, ...theirProps } = props;
let internalTransitionRef = (0, import_react43.useRef)(null);
let transitionRef = useSyncRefs(internalTransitionRef, ref);
useServerHandoffComplete();
let usesOpenClosedState = useOpenClosed();
if (show === void 0 && usesOpenClosedState !== null) {
show = (usesOpenClosedState & 1 /* Open */) === 1 /* Open */;
}
if (![true, false].includes(show)) {
throw new Error("A <Transition /> is used but it is missing a `show={true | false}` prop.");
}
let [state, setState] = (0, import_react43.useState)(show ? "visible" /* Visible */ : "hidden" /* Hidden */);
let nestingBag = useNesting(() => {
setState("hidden" /* Hidden */);
});
let [initial, setInitial] = (0, import_react43.useState)(true);
let changes = (0, import_react43.useRef)([show]);
useIsoMorphicEffect(() => {
if (initial === false) {
return;
}
if (changes.current[changes.current.length - 1] !== show) {
changes.current.push(show);
setInitial(false);
}
}, [changes, show]);
let transitionBag = (0, import_react43.useMemo)(
() => ({ show, appear, initial }),
[show, appear, initial]
);
(0, import_react43.useEffect)(() => {
if (show) {
setState("visible" /* Visible */);
} else if (!hasChildren(nestingBag)) {
setState("hidden" /* Hidden */);
} else if (true) {
let node = internalTransitionRef.current;
if (!node)
return;
let rect = node.getBoundingClientRect();
if (rect.x === 0 && rect.y === 0 && rect.width === 0 && rect.height === 0) {
setState("hidden" /* Hidden */);
}
}
}, [show, nestingBag]);
let sharedProps = { unmount };
let beforeEnter = useEvent(() => {
var _a3;
if (initial)
setInitial(false);
(_a3 = props.beforeEnter) == null ? void 0 : _a3.call(props);
});
let beforeLeave = useEvent(() => {
var _a3;
if (initial)
setInitial(false);
(_a3 = props.beforeLeave) == null ? void 0 : _a3.call(props);
});
return /* @__PURE__ */ import_react43.default.createElement(NestingContext.Provider, { value: nestingBag }, /* @__PURE__ */ import_react43.default.createElement(TransitionContext.Provider, { value: transitionBag }, render({
ourProps: {
...sharedProps,
as: import_react43.Fragment,
children: /* @__PURE__ */ import_react43.default.createElement(
TransitionChild,
{
ref: transitionRef,
...sharedProps,
...theirProps,
beforeEnter,
beforeLeave
}
)
},
theirProps: {},
defaultTag: import_react43.Fragment,
features: TransitionChildRenderFeatures,
visible: state === "visible" /* Visible */,
name: "Transition"
})));
}
function ChildFn(props, ref) {
let hasTransitionContext = (0, import_react43.useContext)(TransitionContext) !== null;
let hasOpenClosedContext = useOpenClosed() !== null;
return /* @__PURE__ */ import_react43.default.createElement(import_react43.default.Fragment, null, !hasTransitionContext && hasOpenClosedContext ? (
// @ts-expect-error This is an object
/* @__PURE__ */ import_react43.default.createElement(TransitionRoot, { ref, ...props })
) : (
// @ts-expect-error This is an object
/* @__PURE__ */ import_react43.default.createElement(TransitionChild, { ref, ...props })
));
}
var TransitionRoot = forwardRefWithAs(
TransitionRootFn
);
var TransitionChild = forwardRefWithAs(
TransitionChildFn
);
var Child = forwardRefWithAs(ChildFn);
var Transition = Object.assign(TransitionRoot, { Child, Root: TransitionRoot });
/*! Bundled license information:
@tanstack/react-virtual/build/lib/_virtual/_rollupPluginBabelHelpers.mjs:
(**
* react-virtual
*
* Copyright (c) TanStack
*
* This source code is licensed under the MIT license found in the
* LICENSE.md file in the root directory of this source tree.
*
* @license MIT
*)
@tanstack/virtual-core/build/lib/_virtual/_rollupPluginBabelHelpers.mjs:
(**
* virtual-core
*
* Copyright (c) TanStack
*
* This source code is licensed under the MIT license found in the
* LICENSE.md file in the root directory of this source tree.
*
* @license MIT
*)
@tanstack/virtual-core/build/lib/utils.mjs:
(**
* virtual-core
*
* Copyright (c) TanStack
*
* This source code is licensed under the MIT license found in the
* LICENSE.md file in the root directory of this source tree.
*
* @license MIT
*)
@tanstack/virtual-core/build/lib/index.mjs:
(**
* virtual-core
*
* Copyright (c) TanStack
*
* This source code is licensed under the MIT license found in the
* LICENSE.md file in the root directory of this source tree.
*
* @license MIT
*)
@tanstack/react-virtual/build/lib/index.mjs:
(**
* react-virtual
*
* Copyright (c) TanStack
*
* This source code is licensed under the MIT license found in the
* LICENSE.md file in the root directory of this source tree.
*
* @license MIT
*)
*/