feat: manually mirror opencoze's code from bytedance

Change-Id: I09a73aadda978ad9511264a756b2ce51f5761adf
This commit is contained in:
fanlv
2025-07-20 17:36:12 +08:00
commit 890153324f
14811 changed files with 1923430 additions and 0 deletions

View File

@@ -0,0 +1,117 @@
/*
* Copyright 2025 coze-dev Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/* eslint-disable complexity */
import { useEffect, useRef, useState, type FC } from 'react';
import classNames from 'classnames';
import { useSize } from 'ahooks';
import { getNumberBetween } from '../../utils';
import styles from './index.module.less';
interface IProps {
left?: number;
top?: number;
offsetY?: number;
offsetX?: number;
children?: React.ReactNode;
position?: 'bottom-center' | 'bottom-right' | 'top-center';
zIndex?: number;
onClick?: (e: React.MouseEvent) => void;
className?: string;
limitRect?: {
width?: number;
height?: number;
};
}
export const PopInScreen: FC<IProps> = props => {
const ref = useRef(null);
const {
left = 0,
top = 0,
children,
position = 'bottom-center',
zIndex = 1000,
onClick,
className,
limitRect,
} = props;
// const documentSize = useSize(document.body);
const childrenSize = useSize(ref.current);
let maxLeft = (limitRect?.width ?? Infinity) - (childrenSize?.width ?? 0) / 2;
let minLeft = (childrenSize?.width ?? 0) / 2;
let transform = 'translate(-50%, 0)';
if (position === 'bottom-right') {
maxLeft = (limitRect?.width ?? Infinity) - (childrenSize?.width ?? 0);
minLeft = 0;
transform = 'translate(0, 0)';
} else if (position === 'top-center') {
transform = 'translate(-50%, -100%)';
}
/**
* ahooks useSize 初次执行会返回 undefined导致组件位置计算错误
* 这里监听 childrenSize ,如果为 undefined 则延迟 100ms 再渲染,以修正组件位置
*/
const [id, setId] = useState('');
const timer = useRef<NodeJS.Timeout>();
useEffect(() => {
clearTimeout(timer.current);
if (!childrenSize) {
timer.current = setTimeout(() => {
setId(`${Math.random()}`);
}, 100);
}
}, [childrenSize]);
return (
<div
ref={ref}
onClick={onClick}
className={classNames([
styles['pop-in-screen'],
'!fixed',
'coz-tooltip semi-tooltip-wrapper',
'p-0',
className,
])}
style={{
left: getNumberBetween({
value: left,
max: maxLeft,
min: minLeft,
}),
top: getNumberBetween({
value: top,
max: (limitRect?.height ?? Infinity) - (childrenSize?.height ?? 0),
min: position === 'top-center' ? (childrenSize?.height ?? 0) : 0,
}),
zIndex,
opacity: 1,
maxWidth: 'unset',
transform,
}}
>
{/* 为了触发二次渲染 */}
<div className="hidden" id={id} />
{children}
</div>
);
};