fix(ui): Pin modal action bar; prevent bottom content overflow\n\n- Move action buttons to fixed .modal-footer at the bottom\n- Make modal a column flex container; scroll only body\n- Ensure buttons remain visible on small viewports\n- Remove sticky edge cases causing leaked content

This commit is contained in:
Jason
2025-08-26 12:34:47 +08:00
parent 57d21fabcf
commit 606ee67778
3 changed files with 125 additions and 41 deletions

2
.gitignore vendored
View File

@@ -7,4 +7,4 @@ release/
.env.local .env.local
*.tsbuildinfo *.tsbuildinfo
.npmrc .npmrc
CLAUDE.md CLAUDE.mdAGENTS.md

View File

@@ -13,20 +13,74 @@
.modal-content { .modal-content {
background: white; background: white;
border-radius: 8px; border-radius: 10px;
padding: 2rem; padding: 0;
width: 90%; width: 90%;
max-width: 600px; max-width: 640px;
max-height: 90vh; max-height: 90vh;
overflow-y: auto; overflow: hidden; /* 由 body 滚动,标题栏固定 */
box-shadow: 0 4px 16px rgba(0, 0, 0, 0.2); box-shadow: 0 16px 40px rgba(0, 0, 0, 0.2);
position: relative; position: relative;
z-index: 1001; z-index: 1001;
display: flex; /* 纵向布局,便于底栏固定 */
flex-direction: column;
} }
.modal-content h2 { /* 模拟窗口标题栏 */
margin-bottom: 1.5rem; .modal-titlebar {
color: #2c3e50; display: flex;
align-items: center;
justify-content: space-between;
height: 3rem; /* 与主窗口标题栏一致 */
padding: 0 12px; /* 接近主头部的水平留白 */
background: #3498db; /* 与 .app-header 相同 */
color: #fff;
border-top-left-radius: 10px;
border-top-right-radius: 10px;
}
/* 左侧占位以保证标题居中(与右侧关闭按钮宽度相当) */
.modal-spacer { width: 32px; flex: 0 0 32px; }
.modal-title {
flex: 1;
text-align: center;
color: #fff;
font-weight: 600;
font-size: 1rem;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.modal-close-btn {
background: transparent;
border: none;
color: #fff;
font-size: 20px;
line-height: 1;
padding: 2px 6px;
border-radius: 6px;
cursor: pointer;
}
.modal-close-btn:hover {
background: rgba(255, 255, 255, 0.18);
color: #fff;
}
.modal-form { /* 表单外层包裹 body + footer */
display: flex;
flex-direction: column;
flex: 1 1 auto;
min-height: 0; /* 允许子元素正确计算高度 */
}
.modal-body {
padding: 1.25rem 1.5rem 1.5rem;
overflow: auto; /* 仅内容区滚动 */
flex: 1 1 auto;
min-height: 0;
} }
.error-message { .error-message {
@@ -109,11 +163,13 @@
border-color: #3498db; border-color: #3498db;
} }
.form-actions { .modal-footer { /* 固定在弹窗底部(非滚动区) */
display: flex; display: flex;
gap: 1rem; gap: 1rem;
justify-content: flex-end; justify-content: flex-end;
margin-top: 2rem; padding: 0.75rem 1.5rem;
border-top: 1px solid #ecf0f1;
background: #fff;
} }
.cancel-btn, .cancel-btn,

View File

@@ -181,34 +181,59 @@ const ProviderForm: React.FC<ProviderFormProps> = ({
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, []); }, []);
// 支持按下 ESC 关闭弹窗
useEffect(() => {
const onKeyDown = (e: KeyboardEvent) => {
if (e.key === 'Escape') {
e.preventDefault();
onClose();
}
};
window.addEventListener('keydown', onKeyDown);
return () => window.removeEventListener('keydown', onKeyDown);
}, [onClose]);
return ( return (
<div className="modal-overlay"> <div className="modal-overlay" onMouseDown={(e) => { if (e.target === e.currentTarget) onClose(); }}>
<div className="modal-content"> <div className="modal-content">
<h2>{title}</h2> <div className="modal-titlebar">
<div className="modal-spacer" />
<div className="modal-title" title={title}>{title}</div>
<button
type="button"
className="modal-close-btn"
aria-label="关闭"
onClick={onClose}
title="关闭"
>
×
</button>
</div>
{error && <div className="error-message">{error}</div>} <form onSubmit={handleSubmit} className="modal-form">
<div className="modal-body">
{error && <div className="error-message">{error}</div>}
{showPresets && ( {showPresets && (
<div className="presets"> <div className="presets">
<label></label> <label></label>
<div className="preset-buttons"> <div className="preset-buttons">
{providerPresets.map((preset, index) => ( {providerPresets.map((preset, index) => (
<button <button
key={index} key={index}
type="button" type="button"
className={`preset-btn ${ className={`preset-btn ${
selectedPreset === index ? "selected" : "" selectedPreset === index ? "selected" : ""
}`} }`}
onClick={() => applyPreset(preset, index)} onClick={() => applyPreset(preset, index)}
> >
{preset.name} {preset.name}
</button> </button>
))} ))}
</div>
</div> </div>
</div> )}
)}
<form onSubmit={handleSubmit}>
<div className="form-group"> <div className="form-group">
<label htmlFor="name"> *</label> <label htmlFor="name"> *</label>
<input <input
@@ -282,14 +307,17 @@ const ProviderForm: React.FC<ProviderFormProps> = ({
</small> </small>
</div> </div>
<div className="form-actions">
<button type="button" className="cancel-btn" onClick={onClose}>
</button>
<button type="submit" className="submit-btn">
{submitText}
</button>
</div> </div>
<div className="modal-footer">
<button type="button" className="cancel-btn" onClick={onClose}>
</button>
<button type="submit" className="submit-btn">
{submitText}
</button>
</div>
</form> </form>
</div> </div>
</div> </div>