Sider.js 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. import _objectSpread from "@babel/runtime/helpers/esm/objectSpread2";
  2. import { createVNode as _createVNode } from "vue";
  3. import classNames from '../_util/classNames';
  4. import { inject, defineComponent, shallowRef, watch, onMounted, onBeforeUnmount, provide } from 'vue';
  5. import PropTypes from '../_util/vue-types';
  6. import { tuple } from '../_util/type';
  7. import initDefaultProps from '../_util/props-util/initDefaultProps';
  8. import isNumeric from '../_util/isNumeric';
  9. import BarsOutlined from "@ant-design/icons-vue/es/icons/BarsOutlined";
  10. import RightOutlined from "@ant-design/icons-vue/es/icons/RightOutlined";
  11. import LeftOutlined from "@ant-design/icons-vue/es/icons/LeftOutlined";
  12. import useConfigInject from '../config-provider/hooks/useConfigInject';
  13. import { SiderCollapsedKey, SiderHookProviderKey } from './injectionKey';
  14. const dimensionMaxMap = {
  15. xs: '479.98px',
  16. sm: '575.98px',
  17. md: '767.98px',
  18. lg: '991.98px',
  19. xl: '1199.98px',
  20. xxl: '1599.98px',
  21. xxxl: '1999.98px'
  22. };
  23. export const siderProps = () => ({
  24. prefixCls: String,
  25. collapsible: {
  26. type: Boolean,
  27. default: undefined
  28. },
  29. collapsed: {
  30. type: Boolean,
  31. default: undefined
  32. },
  33. defaultCollapsed: {
  34. type: Boolean,
  35. default: undefined
  36. },
  37. reverseArrow: {
  38. type: Boolean,
  39. default: undefined
  40. },
  41. zeroWidthTriggerStyle: {
  42. type: Object,
  43. default: undefined
  44. },
  45. trigger: PropTypes.any,
  46. width: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),
  47. collapsedWidth: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),
  48. breakpoint: PropTypes.oneOf(tuple('xs', 'sm', 'md', 'lg', 'xl', 'xxl', 'xxxl')),
  49. theme: PropTypes.oneOf(tuple('light', 'dark')).def('dark'),
  50. onBreakpoint: Function,
  51. onCollapse: Function
  52. });
  53. const generateId = (() => {
  54. let i = 0;
  55. return function () {
  56. let prefix = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : '';
  57. i += 1;
  58. return `${prefix}${i}`;
  59. };
  60. })();
  61. export default defineComponent({
  62. compatConfig: {
  63. MODE: 3
  64. },
  65. name: 'ALayoutSider',
  66. inheritAttrs: false,
  67. props: initDefaultProps(siderProps(), {
  68. collapsible: false,
  69. defaultCollapsed: false,
  70. reverseArrow: false,
  71. width: 200,
  72. collapsedWidth: 80
  73. }),
  74. emits: ['breakpoint', 'update:collapsed', 'collapse'],
  75. setup(props, _ref) {
  76. let {
  77. emit,
  78. attrs,
  79. slots
  80. } = _ref;
  81. const {
  82. prefixCls
  83. } = useConfigInject('layout-sider', props);
  84. const siderHook = inject(SiderHookProviderKey, undefined);
  85. const collapsed = shallowRef(!!(props.collapsed !== undefined ? props.collapsed : props.defaultCollapsed));
  86. const below = shallowRef(false);
  87. watch(() => props.collapsed, () => {
  88. collapsed.value = !!props.collapsed;
  89. });
  90. provide(SiderCollapsedKey, collapsed);
  91. const handleSetCollapsed = (value, type) => {
  92. if (props.collapsed === undefined) {
  93. collapsed.value = value;
  94. }
  95. emit('update:collapsed', value);
  96. emit('collapse', value, type);
  97. };
  98. // ========================= Responsive =========================
  99. const responsiveHandlerRef = shallowRef(mql => {
  100. below.value = mql.matches;
  101. emit('breakpoint', mql.matches);
  102. if (collapsed.value !== mql.matches) {
  103. handleSetCollapsed(mql.matches, 'responsive');
  104. }
  105. });
  106. let mql;
  107. function responsiveHandler(mql) {
  108. return responsiveHandlerRef.value(mql);
  109. }
  110. const uniqueId = generateId('ant-sider-');
  111. siderHook && siderHook.addSider(uniqueId);
  112. onMounted(() => {
  113. watch(() => props.breakpoint, () => {
  114. try {
  115. mql === null || mql === void 0 ? void 0 : mql.removeEventListener('change', responsiveHandler);
  116. } catch (error) {
  117. mql === null || mql === void 0 ? void 0 : mql.removeListener(responsiveHandler);
  118. }
  119. if (typeof window !== 'undefined') {
  120. const {
  121. matchMedia
  122. } = window;
  123. if (matchMedia && props.breakpoint && props.breakpoint in dimensionMaxMap) {
  124. mql = matchMedia(`(max-width: ${dimensionMaxMap[props.breakpoint]})`);
  125. try {
  126. mql.addEventListener('change', responsiveHandler);
  127. } catch (error) {
  128. mql.addListener(responsiveHandler);
  129. }
  130. responsiveHandler(mql);
  131. }
  132. }
  133. }, {
  134. immediate: true
  135. });
  136. });
  137. onBeforeUnmount(() => {
  138. try {
  139. mql === null || mql === void 0 ? void 0 : mql.removeEventListener('change', responsiveHandler);
  140. } catch (error) {
  141. mql === null || mql === void 0 ? void 0 : mql.removeListener(responsiveHandler);
  142. }
  143. siderHook && siderHook.removeSider(uniqueId);
  144. });
  145. const toggle = () => {
  146. handleSetCollapsed(!collapsed.value, 'clickTrigger');
  147. };
  148. return () => {
  149. var _a, _b;
  150. const pre = prefixCls.value;
  151. const {
  152. collapsedWidth,
  153. width,
  154. reverseArrow,
  155. zeroWidthTriggerStyle,
  156. trigger = (_a = slots.trigger) === null || _a === void 0 ? void 0 : _a.call(slots),
  157. collapsible,
  158. theme
  159. } = props;
  160. const rawWidth = collapsed.value ? collapsedWidth : width;
  161. // use "px" as fallback unit for width
  162. const siderWidth = isNumeric(rawWidth) ? `${rawWidth}px` : String(rawWidth);
  163. // special trigger when collapsedWidth == 0
  164. const zeroWidthTrigger = parseFloat(String(collapsedWidth || 0)) === 0 ? _createVNode("span", {
  165. "onClick": toggle,
  166. "class": classNames(`${pre}-zero-width-trigger`, `${pre}-zero-width-trigger-${reverseArrow ? 'right' : 'left'}`),
  167. "style": zeroWidthTriggerStyle
  168. }, [trigger || _createVNode(BarsOutlined, null, null)]) : null;
  169. const iconObj = {
  170. expanded: reverseArrow ? _createVNode(RightOutlined, null, null) : _createVNode(LeftOutlined, null, null),
  171. collapsed: reverseArrow ? _createVNode(LeftOutlined, null, null) : _createVNode(RightOutlined, null, null)
  172. };
  173. const status = collapsed.value ? 'collapsed' : 'expanded';
  174. const defaultTrigger = iconObj[status];
  175. const triggerDom = trigger !== null ? zeroWidthTrigger || _createVNode("div", {
  176. "class": `${pre}-trigger`,
  177. "onClick": toggle,
  178. "style": {
  179. width: siderWidth
  180. }
  181. }, [trigger || defaultTrigger]) : null;
  182. const divStyle = [attrs.style, {
  183. flex: `0 0 ${siderWidth}`,
  184. maxWidth: siderWidth,
  185. minWidth: siderWidth,
  186. width: siderWidth
  187. }];
  188. const siderCls = classNames(pre, `${pre}-${theme}`, {
  189. [`${pre}-collapsed`]: !!collapsed.value,
  190. [`${pre}-has-trigger`]: collapsible && trigger !== null && !zeroWidthTrigger,
  191. [`${pre}-below`]: !!below.value,
  192. [`${pre}-zero-width`]: parseFloat(siderWidth) === 0
  193. }, attrs.class);
  194. return _createVNode("aside", _objectSpread(_objectSpread({}, attrs), {}, {
  195. "class": siderCls,
  196. "style": divStyle
  197. }), [_createVNode("div", {
  198. "class": `${pre}-children`
  199. }, [(_b = slots.default) === null || _b === void 0 ? void 0 : _b.call(slots)]), collapsible || below.value && zeroWidthTrigger ? triggerDom : null]);
  200. };
  201. }
  202. });