mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-04 18:59:08 +01:00
fix dynamic setting of suggestions modal (#272)
This commit is contained in:
parent
cefc491bf2
commit
7d98aeb2c9
@ -34,6 +34,7 @@
|
|||||||
border: 0.5px solid var(--modal-border-color);
|
border: 0.5px solid var(--modal-border-color);
|
||||||
background: var(--modal-bg-color);
|
background: var(--modal-bg-color);
|
||||||
box-shadow: 0px 8px 32px 0px rgba(0, 0, 0, 0.25);
|
box-shadow: 0px 8px 32px 0px rgba(0, 0, 0, 0.25);
|
||||||
|
height: auto;
|
||||||
|
|
||||||
.modal-close-btn {
|
.modal-close-btn {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
|
@ -2,7 +2,7 @@ import { Input } from "@/app/element/input";
|
|||||||
import { InputDecoration } from "@/app/element/inputdecoration";
|
import { InputDecoration } from "@/app/element/inputdecoration";
|
||||||
import { useDimensions } from "@/app/hook/useDimensions";
|
import { useDimensions } from "@/app/hook/useDimensions";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import React, { useEffect, useRef, useState } from "react";
|
import React, { forwardRef, useEffect, useRef, useState } from "react";
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
|
|
||||||
import "./typeaheadmodal.less";
|
import "./typeaheadmodal.less";
|
||||||
@ -81,15 +81,9 @@ interface SuggestionsProps {
|
|||||||
onSelect?: (_: string) => void;
|
onSelect?: (_: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
function Suggestions({ suggestions, onSelect }: SuggestionsProps) {
|
const Suggestions = forwardRef<HTMLDivElement, SuggestionsProps>(({ suggestions, onSelect }: SuggestionsProps, ref) => {
|
||||||
const suggestionsWrapperRef = useRef<HTMLDivElement>(null);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div ref={ref} className="suggestions-wrapper" style={{ marginTop: suggestions?.length > 0 ? "8px" : "0" }}>
|
||||||
ref={suggestionsWrapperRef}
|
|
||||||
className="suggestions-wrapper"
|
|
||||||
style={{ marginTop: suggestions?.length > 0 ? "8px" : "0" }}
|
|
||||||
>
|
|
||||||
{suggestions?.map((suggestion, index) => (
|
{suggestions?.map((suggestion, index) => (
|
||||||
<div className="suggestion" key={index} onClick={() => onSelect(suggestion.value)}>
|
<div className="suggestion" key={index} onClick={() => onSelect(suggestion.value)}>
|
||||||
{suggestion.label}
|
{suggestion.label}
|
||||||
@ -97,7 +91,7 @@ function Suggestions({ suggestions, onSelect }: SuggestionsProps) {
|
|||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
});
|
||||||
|
|
||||||
interface TypeAheadModalProps {
|
interface TypeAheadModalProps {
|
||||||
anchor: React.MutableRefObject<HTMLDivElement>;
|
anchor: React.MutableRefObject<HTMLDivElement>;
|
||||||
@ -125,20 +119,31 @@ const TypeAheadModal = ({
|
|||||||
const { width, height } = useDimensions(anchor);
|
const { width, height } = useDimensions(anchor);
|
||||||
const modalRef = useRef<HTMLDivElement>(null);
|
const modalRef = useRef<HTMLDivElement>(null);
|
||||||
const inputRef = useRef<HTMLDivElement>(null);
|
const inputRef = useRef<HTMLDivElement>(null);
|
||||||
|
const suggestionsWrapperRef = useRef<HTMLDivElement>(null);
|
||||||
const [suggestionsHeight, setSuggestionsHeight] = useState<number | undefined>(undefined);
|
const [suggestionsHeight, setSuggestionsHeight] = useState<number | undefined>(undefined);
|
||||||
|
const [modalHeight, setModalHeight] = useState<string | undefined>(undefined);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (modalRef.current && inputRef.current) {
|
if (modalRef.current && inputRef.current && suggestionsWrapperRef.current) {
|
||||||
const modalHeight = modalRef.current.getBoundingClientRect().height;
|
const modalPadding = 32;
|
||||||
const inputHeight = inputRef.current.getBoundingClientRect().height;
|
const inputHeight = inputRef.current.getBoundingClientRect().height;
|
||||||
|
let suggestionsTotalHeight = 0;
|
||||||
|
|
||||||
// Get the padding value (assuming padding is uniform on all sides)
|
const suggestionItems = suggestionsWrapperRef.current.children;
|
||||||
const padding = 16 * 2; // 16px top + 16px bottom
|
for (let i = 0; i < suggestionItems.length; i++) {
|
||||||
|
suggestionsTotalHeight += suggestionItems[i].getBoundingClientRect().height;
|
||||||
// Subtract the input height and padding from the modal height
|
|
||||||
setSuggestionsHeight(modalHeight - inputHeight - padding);
|
|
||||||
}
|
}
|
||||||
}, [width, height]);
|
|
||||||
|
const totalHeight = modalPadding + inputHeight + suggestionsTotalHeight;
|
||||||
|
const maxHeight = height * 0.8;
|
||||||
|
const computedHeight = totalHeight > maxHeight ? maxHeight : totalHeight;
|
||||||
|
|
||||||
|
setModalHeight(`${computedHeight}px`);
|
||||||
|
|
||||||
|
const padding = 16 * 2;
|
||||||
|
setSuggestionsHeight(computedHeight - inputHeight - padding);
|
||||||
|
}
|
||||||
|
}, [height, suggestions.length]);
|
||||||
|
|
||||||
const renderBackdrop = (onClick) => <div className="type-ahead-modal-backdrop" onClick={onClick}></div>;
|
const renderBackdrop = (onClick) => <div className="type-ahead-modal-backdrop" onClick={onClick}></div>;
|
||||||
|
|
||||||
@ -162,7 +167,7 @@ const TypeAheadModal = ({
|
|||||||
className={clsx("type-ahead-modal", className)}
|
className={clsx("type-ahead-modal", className)}
|
||||||
style={{
|
style={{
|
||||||
width: width * 0.6,
|
width: width * 0.6,
|
||||||
maxHeight: height * 0.8,
|
maxHeight: modalHeight,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div className="content-wrapper">
|
<div className="content-wrapper">
|
||||||
@ -187,7 +192,7 @@ const TypeAheadModal = ({
|
|||||||
overflowY: "auto",
|
overflowY: "auto",
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Suggestions suggestions={suggestions} onSelect={handleSelect} />
|
<Suggestions ref={suggestionsWrapperRef} suggestions={suggestions} onSelect={handleSelect} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -687,7 +687,6 @@ function PreviewView({ blockId, model }: { blockId: string; model: PreviewModel
|
|||||||
anchor={contentRef}
|
anchor={contentRef}
|
||||||
onKeyDown={(e) => keyutil.keydownWrapper(handleKeyDown)(e)}
|
onKeyDown={(e) => keyutil.keydownWrapper(handleKeyDown)(e)}
|
||||||
onSelect={handleFileSuggestionSelect}
|
onSelect={handleFileSuggestionSelect}
|
||||||
onSubmit={(value) => console.log(value)}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<div
|
<div
|
||||||
|
Loading…
Reference in New Issue
Block a user