OptionList.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. "use strict";
  2. var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
  3. Object.defineProperty(exports, "__esModule", {
  4. value: true
  5. });
  6. exports.default = void 0;
  7. var _vue = require("vue");
  8. var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
  9. var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties"));
  10. var _objectSpread2 = _interopRequireDefault(require("@babel/runtime/helpers/objectSpread2"));
  11. var _TransBtn = _interopRequireDefault(require("./TransBtn"));
  12. var _KeyCode = _interopRequireDefault(require("../_util/KeyCode"));
  13. var _classNames2 = _interopRequireDefault(require("../_util/classNames"));
  14. var _pickAttrs = _interopRequireDefault(require("../_util/pickAttrs"));
  15. var _propsUtil = require("../_util/props-util");
  16. var _createRef = _interopRequireDefault(require("../_util/createRef"));
  17. var _vcVirtualList = _interopRequireDefault(require("../vc-virtual-list"));
  18. var _useMemo = _interopRequireDefault(require("../_util/hooks/useMemo"));
  19. var _platformUtil = require("./utils/platformUtil");
  20. var _omit = _interopRequireDefault(require("../_util/omit"));
  21. var _useBaseProps = _interopRequireDefault(require("./hooks/useBaseProps"));
  22. var _SelectContext = _interopRequireDefault(require("./SelectContext"));
  23. var _excluded = ["disabled", "title", "children", "style", "class", "className"];
  24. function isTitleType(content) {
  25. return typeof content === 'string' || typeof content === 'number';
  26. }
  27. /**
  28. * Using virtual list of option display.
  29. * Will fallback to dom if use customize render.
  30. */
  31. var OptionList = (0, _vue.defineComponent)({
  32. compatConfig: {
  33. MODE: 3
  34. },
  35. name: 'OptionList',
  36. inheritAttrs: false,
  37. slots: ['option'],
  38. setup: function setup(_, _ref) {
  39. var expose = _ref.expose,
  40. slots = _ref.slots;
  41. var baseProps = (0, _useBaseProps.default)();
  42. var props = (0, _SelectContext.default)();
  43. var itemPrefixCls = (0, _vue.computed)(function () {
  44. return "".concat(baseProps.prefixCls, "-item");
  45. });
  46. var memoFlattenOptions = (0, _useMemo.default)(function () {
  47. return props.flattenOptions;
  48. }, [function () {
  49. return baseProps.open;
  50. }, function () {
  51. return props.flattenOptions;
  52. }], function (next) {
  53. return next[0];
  54. });
  55. // =========================== List ===========================
  56. var listRef = (0, _createRef.default)();
  57. var onListMouseDown = function onListMouseDown(event) {
  58. event.preventDefault();
  59. };
  60. var scrollIntoView = function scrollIntoView(args) {
  61. if (listRef.current) {
  62. listRef.current.scrollTo(typeof args === 'number' ? {
  63. index: args
  64. } : args);
  65. }
  66. };
  67. // ========================== Active ==========================
  68. var getEnabledActiveIndex = function getEnabledActiveIndex(index) {
  69. var offset = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 1;
  70. var len = memoFlattenOptions.value.length;
  71. for (var i = 0; i < len; i += 1) {
  72. var current = (index + i * offset + len) % len;
  73. var _memoFlattenOptions$v = memoFlattenOptions.value[current],
  74. group = _memoFlattenOptions$v.group,
  75. data = _memoFlattenOptions$v.data;
  76. if (!group && !data.disabled) {
  77. return current;
  78. }
  79. }
  80. return -1;
  81. };
  82. var state = (0, _vue.reactive)({
  83. activeIndex: getEnabledActiveIndex(0)
  84. });
  85. var setActive = function setActive(index) {
  86. var fromKeyboard = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;
  87. state.activeIndex = index;
  88. var info = {
  89. source: fromKeyboard ? 'keyboard' : 'mouse'
  90. };
  91. // Trigger active event
  92. var flattenItem = memoFlattenOptions.value[index];
  93. if (!flattenItem) {
  94. props.onActiveValue(null, -1, info);
  95. return;
  96. }
  97. props.onActiveValue(flattenItem.value, index, info);
  98. };
  99. // Auto active first item when list length or searchValue changed
  100. (0, _vue.watch)([function () {
  101. return memoFlattenOptions.value.length;
  102. }, function () {
  103. return baseProps.searchValue;
  104. }], function () {
  105. setActive(props.defaultActiveFirstOption !== false ? getEnabledActiveIndex(0) : -1);
  106. }, {
  107. immediate: true
  108. });
  109. // https://github.com/ant-design/ant-design/issues/34975
  110. var isSelected = function isSelected(value) {
  111. return props.rawValues.has(value) && baseProps.mode !== 'combobox';
  112. };
  113. // Auto scroll to item position in single mode
  114. (0, _vue.watch)([function () {
  115. return baseProps.open;
  116. }, function () {
  117. return baseProps.searchValue;
  118. }], function () {
  119. if (!baseProps.multiple && baseProps.open && props.rawValues.size === 1) {
  120. var value = Array.from(props.rawValues)[0];
  121. var index = (0, _vue.toRaw)(memoFlattenOptions.value).findIndex(function (_ref2) {
  122. var data = _ref2.data;
  123. return data[props.fieldNames.value] === value;
  124. });
  125. if (index !== -1) {
  126. setActive(index);
  127. (0, _vue.nextTick)(function () {
  128. scrollIntoView(index);
  129. });
  130. }
  131. }
  132. // Force trigger scrollbar visible when open
  133. if (baseProps.open) {
  134. (0, _vue.nextTick)(function () {
  135. var _listRef$current;
  136. (_listRef$current = listRef.current) === null || _listRef$current === void 0 ? void 0 : _listRef$current.scrollTo(undefined);
  137. });
  138. }
  139. }, {
  140. immediate: true,
  141. flush: 'post'
  142. });
  143. // ========================== Values ==========================
  144. var onSelectValue = function onSelectValue(value) {
  145. if (value !== undefined) {
  146. props.onSelect(value, {
  147. selected: !props.rawValues.has(value)
  148. });
  149. }
  150. // Single mode should always close by select
  151. if (!baseProps.multiple) {
  152. baseProps.toggleOpen(false);
  153. }
  154. };
  155. var getLabel = function getLabel(item) {
  156. return typeof item.label === 'function' ? item.label() : item.label;
  157. };
  158. function renderItem(index) {
  159. var item = memoFlattenOptions.value[index];
  160. if (!item) return null;
  161. var itemData = item.data || {};
  162. var value = itemData.value;
  163. var group = item.group;
  164. var attrs = (0, _pickAttrs.default)(itemData, true);
  165. var mergedLabel = getLabel(item);
  166. return item ? (0, _vue.createVNode)("div", (0, _objectSpread2.default)((0, _objectSpread2.default)({
  167. "aria-label": typeof mergedLabel === 'string' && !group ? mergedLabel : null
  168. }, attrs), {}, {
  169. "key": index,
  170. "role": group ? 'presentation' : 'option',
  171. "id": "".concat(baseProps.id, "_list_").concat(index),
  172. "aria-selected": isSelected(value)
  173. }), [value]) : null;
  174. }
  175. var onKeydown = function onKeydown(event) {
  176. var which = event.which,
  177. ctrlKey = event.ctrlKey;
  178. switch (which) {
  179. // >>> Arrow keys & ctrl + n/p on Mac
  180. case _KeyCode.default.N:
  181. case _KeyCode.default.P:
  182. case _KeyCode.default.UP:
  183. case _KeyCode.default.DOWN:
  184. {
  185. var offset = 0;
  186. if (which === _KeyCode.default.UP) {
  187. offset = -1;
  188. } else if (which === _KeyCode.default.DOWN) {
  189. offset = 1;
  190. } else if ((0, _platformUtil.isPlatformMac)() && ctrlKey) {
  191. if (which === _KeyCode.default.N) {
  192. offset = 1;
  193. } else if (which === _KeyCode.default.P) {
  194. offset = -1;
  195. }
  196. }
  197. if (offset !== 0) {
  198. var nextActiveIndex = getEnabledActiveIndex(state.activeIndex + offset, offset);
  199. scrollIntoView(nextActiveIndex);
  200. setActive(nextActiveIndex, true);
  201. }
  202. break;
  203. }
  204. // >>> Select
  205. case _KeyCode.default.ENTER:
  206. {
  207. // value
  208. var item = memoFlattenOptions.value[state.activeIndex];
  209. if (item && !item.data.disabled) {
  210. onSelectValue(item.value);
  211. } else {
  212. onSelectValue(undefined);
  213. }
  214. if (baseProps.open) {
  215. event.preventDefault();
  216. }
  217. break;
  218. }
  219. // >>> Close
  220. case _KeyCode.default.ESC:
  221. {
  222. baseProps.toggleOpen(false);
  223. if (baseProps.open) {
  224. event.stopPropagation();
  225. }
  226. }
  227. }
  228. };
  229. var onKeyup = function onKeyup() {};
  230. var scrollTo = function scrollTo(index) {
  231. scrollIntoView(index);
  232. };
  233. expose({
  234. onKeydown: onKeydown,
  235. onKeyup: onKeyup,
  236. scrollTo: scrollTo
  237. });
  238. return function () {
  239. // const {
  240. // renderItem,
  241. // listRef,
  242. // onListMouseDown,
  243. // itemPrefixCls,
  244. // setActive,
  245. // onSelectValue,
  246. // memoFlattenOptions,
  247. // $slots,
  248. // } = this as any;
  249. var id = baseProps.id,
  250. notFoundContent = baseProps.notFoundContent,
  251. onPopupScroll = baseProps.onPopupScroll;
  252. var menuItemSelectedIcon = props.menuItemSelectedIcon,
  253. fieldNames = props.fieldNames,
  254. virtual = props.virtual,
  255. listHeight = props.listHeight,
  256. listItemHeight = props.listItemHeight;
  257. var renderOption = slots.option;
  258. var activeIndex = state.activeIndex;
  259. var omitFieldNameList = Object.keys(fieldNames).map(function (key) {
  260. return fieldNames[key];
  261. });
  262. // ========================== Render ==========================
  263. if (memoFlattenOptions.value.length === 0) {
  264. return (0, _vue.createVNode)("div", {
  265. "role": "listbox",
  266. "id": "".concat(id, "_list"),
  267. "class": "".concat(itemPrefixCls.value, "-empty"),
  268. "onMousedown": onListMouseDown
  269. }, [notFoundContent]);
  270. }
  271. return (0, _vue.createVNode)(_vue.Fragment, null, [(0, _vue.createVNode)("div", {
  272. "role": "listbox",
  273. "id": "".concat(id, "_list"),
  274. "style": {
  275. height: 0,
  276. width: 0,
  277. overflow: 'hidden'
  278. }
  279. }, [renderItem(activeIndex - 1), renderItem(activeIndex), renderItem(activeIndex + 1)]), (0, _vue.createVNode)(_vcVirtualList.default, {
  280. "itemKey": "key",
  281. "ref": listRef,
  282. "data": memoFlattenOptions.value,
  283. "height": listHeight,
  284. "itemHeight": listItemHeight,
  285. "fullHeight": false,
  286. "onMousedown": onListMouseDown,
  287. "onScroll": onPopupScroll,
  288. "virtual": virtual
  289. }, {
  290. default: function _default(item, itemIndex) {
  291. var _classNames;
  292. var group = item.group,
  293. groupOption = item.groupOption,
  294. data = item.data,
  295. value = item.value;
  296. var key = data.key;
  297. var label = typeof item.label === 'function' ? item.label() : item.label;
  298. // Group
  299. if (group) {
  300. var _data$title;
  301. var groupTitle = (_data$title = data.title) !== null && _data$title !== void 0 ? _data$title : isTitleType(label) && label;
  302. return (0, _vue.createVNode)("div", {
  303. "class": (0, _classNames2.default)(itemPrefixCls.value, "".concat(itemPrefixCls.value, "-group")),
  304. "title": groupTitle
  305. }, [renderOption ? renderOption(data) : label !== undefined ? label : key]);
  306. }
  307. var disabled = data.disabled,
  308. title = data.title,
  309. children = data.children,
  310. style = data.style,
  311. cls = data.class,
  312. className = data.className,
  313. otherProps = (0, _objectWithoutProperties2.default)(data, _excluded);
  314. var passedProps = (0, _omit.default)(otherProps, omitFieldNameList);
  315. // Option
  316. var selected = isSelected(value);
  317. var optionPrefixCls = "".concat(itemPrefixCls.value, "-option");
  318. var optionClassName = (0, _classNames2.default)(itemPrefixCls.value, optionPrefixCls, cls, className, (_classNames = {}, (0, _defineProperty2.default)(_classNames, "".concat(optionPrefixCls, "-grouped"), groupOption), (0, _defineProperty2.default)(_classNames, "".concat(optionPrefixCls, "-active"), activeIndex === itemIndex && !disabled), (0, _defineProperty2.default)(_classNames, "".concat(optionPrefixCls, "-disabled"), disabled), (0, _defineProperty2.default)(_classNames, "".concat(optionPrefixCls, "-selected"), selected), _classNames));
  319. var mergedLabel = getLabel(item);
  320. var iconVisible = !menuItemSelectedIcon || typeof menuItemSelectedIcon === 'function' || selected;
  321. // https://github.com/ant-design/ant-design/issues/34145
  322. var content = typeof mergedLabel === 'number' ? mergedLabel : mergedLabel || value;
  323. // https://github.com/ant-design/ant-design/issues/26717
  324. var optionTitle = isTitleType(content) ? content.toString() : undefined;
  325. if (title !== undefined) {
  326. optionTitle = title;
  327. }
  328. return (0, _vue.createVNode)("div", (0, _objectSpread2.default)((0, _objectSpread2.default)({}, passedProps), {}, {
  329. "aria-selected": selected,
  330. "class": optionClassName,
  331. "title": optionTitle,
  332. "onMousemove": function onMousemove(e) {
  333. if (otherProps.onMousemove) {
  334. otherProps.onMousemove(e);
  335. }
  336. if (activeIndex === itemIndex || disabled) {
  337. return;
  338. }
  339. setActive(itemIndex);
  340. },
  341. "onClick": function onClick(e) {
  342. if (!disabled) {
  343. onSelectValue(value);
  344. }
  345. if (otherProps.onClick) {
  346. otherProps.onClick(e);
  347. }
  348. },
  349. "style": style
  350. }), [(0, _vue.createVNode)("div", {
  351. "class": "".concat(optionPrefixCls, "-content")
  352. }, [renderOption ? renderOption(data) : content]), (0, _propsUtil.isValidElement)(menuItemSelectedIcon) || selected, iconVisible && (0, _vue.createVNode)(_TransBtn.default, {
  353. "class": "".concat(itemPrefixCls.value, "-option-state"),
  354. "customizeIcon": menuItemSelectedIcon,
  355. "customizeIconProps": {
  356. isSelected: selected
  357. }
  358. }, {
  359. default: function _default() {
  360. return [selected ? '✓' : null];
  361. }
  362. })]);
  363. }
  364. })]);
  365. };
  366. }
  367. });
  368. var _default2 = OptionList;
  369. exports.default = _default2;