ant-design-vue/site/src/components/antdv-token-previewer/ThemeSelect.tsx

241 lines
7.1 KiB
Vue

import type { PropType } from 'vue';
import { defineComponent, toRefs, computed } from 'vue';
import makeStyle from './utils/makeStyle';
import classNames from 'ant-design-vue/es/_util/classNames';
import { Button, Dropdown, Menu } from 'ant-design-vue';
import { CloseOutlined, PlusOutlined } from '@ant-design/icons-vue';
import type { Theme } from './interface';
interface ThemeItem extends Theme {
icon?: any;
closable?: boolean;
fixed?: boolean;
}
export type ThemeSelectProps = {
onEnabledThemeChange: (themes: string[]) => void;
onShownThemeChange: (
themes: string[],
selectTheme: string,
info: { type: 'select' | 'deselect' },
) => void;
enabledThemes: string[];
shownThemes: string[];
themes: ThemeItem[];
showAddTheme?: boolean;
};
const useStyle = makeStyle('ThemeSelect', token => ({
'.previewer-theme-select': {
padding: `${token.paddingXXS}px ${token.paddingXS}px`,
borderRadius: 4,
backgroundColor: 'rgba(0, 0, 0, 0.02)',
height: token.controlHeight,
display: 'flex',
alignItems: 'center',
overflow: 'hidden',
[`${token.rootCls}-btn.previewer-theme-select-add-btn`]: {
minWidth: 0,
width: 16,
height: 16,
fontSize: 8,
display: 'inline-flex',
alignItems: 'center',
justifyContent: 'center',
marginInlineStart: token.marginSM,
boxShadow: 'none',
},
'.previewer-theme-select-tag': {
height: 22,
display: 'inline-flex',
alignItems: 'center',
justifyContent: 'center',
boxSizing: 'border-box',
borderRadius: 4,
backgroundColor: token.colorBgContainer,
border: `${token.lineWidth}px ${token.lineType} ${token.colorBorder}`,
paddingInline: 10,
fontSize: token.fontSizeSM,
position: 'relative',
cursor: 'pointer',
// transition: `all ${token.motionDurationMid}`,
'&:not(:last-child)': {
marginInlineEnd: token.marginXS,
},
'&.previewer-theme-select-tag-active': {
border: `${token.lineWidth}px ${token.lineType} ${token['blue-1']}`,
backgroundColor: 'rgba(22,119,255,0.10)',
color: token.colorPrimary,
'&::after': {
content: '""',
borderStartEndRadius: 2,
position: 'absolute',
insetInlineEnd: 2,
top: 2,
width: 6,
height: 6,
background: `linear-gradient(to right top, transparent, transparent 50%, ${token.colorPrimary} 50%, ${token.colorPrimary} 100%)`,
},
},
'.previewer-theme-select-tag-close-btn': {
position: 'absolute',
top: -2,
insetInlineEnd: -2,
width: 12,
height: 12,
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
background: token.colorBgContainer,
boxShadow:
'0 2px 8px -2px rgba(0,0,0,0.05), 0 1px 4px -1px rgba(25,15,15,0.07), 0 0 1px 0 rgba(0,0,0,0.08)',
borderRadius: '50%',
opacity: 0,
pointerEvents: 'none',
zIndex: 2,
color: token.colorIcon,
'> .anticon': {
fontSize: 6,
},
},
'&:hover': {
'.previewer-theme-select-tag-close-btn': {
opacity: 1,
pointerEvents: 'initial',
},
},
},
},
'.previewer-theme-select-dropdown': {
'.previewer-theme-select-dropdown-title': {
[`${token.rootCls}-dropdown-menu-item-group-title`]: {
fontSize: token.fontSizeSM,
paddingBottom: token.padding,
paddingTop: 10,
},
},
},
}));
const ThemeSelect = defineComponent({
name: 'ThemeSelect',
inheritAttrs: false,
props: {
onEnabledThemeChange: { type: Function as PropType<(themes: string[]) => void> },
onShownThemeChange: {
type: Function as PropType<
(themes: string[], selectTheme: string, info: { type: 'select' | 'deselect' }) => void
>,
},
enabledThemes: { type: Array as PropType<string[]> },
shownThemes: { type: Array as PropType<string[]> },
themes: { type: Array as PropType<ThemeItem[]> },
showAddTheme: { type: Boolean },
},
setup(props, { attrs }) {
const { enabledThemes, shownThemes, themes, showAddTheme } = toRefs(props);
const [wrapSSR, hashId] = useStyle();
const dropdownItems = computed(() => [
{
disabled: true,
label: '添加主题即可预览',
className: 'previewer-theme-select-dropdown-title',
type: 'group',
key: 'add-theme-to-preview',
},
...themes.value
.filter(theme => !shownThemes.value.includes(theme.key))
.map(theme => ({
icon: theme.icon,
value: theme.key,
label: theme.name,
key: theme.key,
onClick: () => {
props.onShownThemeChange([...shownThemes.value, theme.key], theme.key, {
type: 'select',
});
},
})),
]);
const shownThemeEntities = computed(() =>
themes.value.filter(theme => shownThemes.value.includes(theme.key)),
);
return () => {
return wrapSSR(
<div {...attrs} class={classNames('previewer-theme-select', hashId.value, attrs.class)}>
{shownThemeEntities.value.map(theme => (
<span
onClick={() => {
if (theme.fixed) {
return;
}
props.onEnabledThemeChange(
enabledThemes.value.includes(theme.key)
? enabledThemes.value.filter(item => item !== theme.key)
: [...enabledThemes.value, theme.key],
);
}}
key={theme.key}
class={classNames('previewer-theme-select-tag', {
'previewer-theme-select-tag-active': enabledThemes.value.includes(theme.key),
})}
>
{theme.name}
{theme.closable && (
<span
class="previewer-theme-select-tag-close-btn"
onClick={e => {
e.stopPropagation();
props.onEnabledThemeChange(
enabledThemes.value.filter(item => item !== theme.key),
);
props.onShownThemeChange(
shownThemes.value.filter(item => item !== theme.key),
theme.key,
{ type: 'deselect' },
);
}}
>
<CloseOutlined />
</span>
)}
</span>
))}
{showAddTheme.value && (
<Dropdown
placement="bottomRight"
trigger={['click']}
overlayClassName={classNames('previewer-theme-select-dropdown', hashId.value)}
v-slots={{
overlay: () => <Menu items={dropdownItems.value} />,
}}
>
<Button
type="primary"
shape="circle"
class="previewer-theme-select-add-btn"
icon={<PlusOutlined />}
/>
</Dropdown>
)}
</div>,
);
};
},
});
export default ThemeSelect;