cascader.test.js 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152
  1. import { Cascader, Icon } from '../../index';
  2. import { clear } from 'jest-date-mock';
  3. import { BASE_CLASS_PREFIX } from '../../../semi-foundation/base/constants';
  4. const animationMs = 200;
  5. const sleep = (ms = animationMs) =>
  6. new Promise((resolve, reject) =>
  7. setTimeout(() => {
  8. resolve(ms);
  9. }, ms)
  10. );
  11. const getPopupContainer = () => document.querySelector(`.${BASE_CLASS_PREFIX}-cascader`);
  12. const treeData = [
  13. {
  14. label: '亚洲',
  15. value: 'Yazhou',
  16. key: 'yazhou',
  17. children: [
  18. {
  19. label: '中国',
  20. value: 'Zhongguo',
  21. key: 'zhongguo',
  22. children: [
  23. {
  24. label: '北京',
  25. value: 'Beijing',
  26. key: 'beijing',
  27. },
  28. {
  29. label: '上海',
  30. value: 'Shanghai',
  31. key: 'shanghai',
  32. },
  33. ],
  34. },
  35. ],
  36. },
  37. {
  38. label: '北美洲',
  39. value: 'Beimeizhou',
  40. key: 'beimeizhou',
  41. children: [
  42. {
  43. label: '美国',
  44. value: 'Meiguo',
  45. key: 'meiguo',
  46. },
  47. {
  48. label: '加拿大',
  49. value: 'Jianada',
  50. key: 'jianada',
  51. },
  52. ],
  53. },
  54. ];
  55. const treeDataWithDisabled = [
  56. {
  57. label: '亚洲',
  58. value: 'Yazhou',
  59. key: 'yazhou',
  60. children: [
  61. {
  62. label: '中国',
  63. value: 'Zhongguo',
  64. key: 'zhongguo',
  65. disabled: true,
  66. children: [
  67. {
  68. label: '北京',
  69. value: 'Beijing',
  70. key: 'beijing',
  71. },
  72. {
  73. label: '上海',
  74. value: 'Shanghai',
  75. key: 'shanghai',
  76. },
  77. ],
  78. },
  79. {
  80. label: '韩国',
  81. value: 'Hanguo',
  82. key: 'hanguo',
  83. }
  84. ],
  85. }
  86. ];
  87. let commonProps = {
  88. // Cascader use Popup Layer to show candidate option,
  89. // but all Popup Layer which extends from Tooltip (eg Popover, Dropdown) have animation and delay.
  90. // Turn off animation and delay during testing, to avoid wating (something like setTimeOut/balabala...) in the test code
  91. motion: false,
  92. mouseEnterDelay: 0,
  93. mouseLeaveDelay: 0,
  94. };
  95. function render(props) {
  96. props = { treeData: treeData, ...commonProps, ...props };
  97. return mount(<Cascader {...props} />, {
  98. attachTo: document.getElementById('container'),
  99. });
  100. }
  101. describe('Cascader', () => {
  102. beforeEach(() => {
  103. // Avoid `attachTo: document.body` Warning
  104. // clear();
  105. const div = document.createElement('div');
  106. div.setAttribute('id', 'container');
  107. document.body.appendChild(div);
  108. });
  109. afterEach(() => {
  110. const div = document.getElementById('container');
  111. if (div) {
  112. document.body.removeChild(div);
  113. }
  114. document.body.innerHTML = '';
  115. });
  116. it('className / style', () => {
  117. let props = {
  118. className: 'test',
  119. style: { height: 420 },
  120. };
  121. let cascader = render(props);
  122. expect(cascader.hasClass('test')).toEqual(true);
  123. expect(cascader.find('div.test')).toHaveStyle('height', 420);
  124. });
  125. it('with placeholder', () => {
  126. const props = { placeholder: 'semi' };
  127. const cascader = render(props);
  128. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection-placeholder`).instance().textContent).toEqual(
  129. 'semi'
  130. );
  131. });
  132. it('with validateStatus', () => {
  133. const props = {};
  134. const cascader = render(props);
  135. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-error`)).toEqual(false);
  136. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-warning`)).toEqual(false);
  137. cascader.setProps({ validateStatus: `error` });
  138. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-error`)).toEqual(true);
  139. cascader.setProps({ validateStatus: `warning` });
  140. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-warning`)).toEqual(true);
  141. });
  142. it('different size', () => {
  143. const props = {};
  144. const cascader = render(props);
  145. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-large`)).toEqual(false);
  146. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-small`)).toEqual(false);
  147. cascader.setProps({ size: 'large' });
  148. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-large`)).toEqual(true);
  149. cascader.setProps({ size: 'small' });
  150. expect(cascader.exists(`.${BASE_CLASS_PREFIX}-cascader-small`)).toEqual(true);
  151. });
  152. it('custom prefix / suffix / insetLabel', () => {
  153. let prefix = <div className="prefix">prefix content</div>;
  154. let suffix = <div className="suffix">suffix content</div>;
  155. let insetLabel = 'semi';
  156. const props = {
  157. prefix: prefix,
  158. suffix: suffix,
  159. };
  160. let cascader = render(props);
  161. expect(cascader.contains(prefix)).toEqual(true);
  162. expect(cascader.contains(suffix)).toEqual(true);
  163. cascader.unmount();
  164. let cascader2 = render({ insetLabel: insetLabel });
  165. expect(cascader2.contains(insetLabel)).toEqual(true);
  166. cascader2.unmount();
  167. });
  168. it('empty data / empty content', () => {
  169. let cascader = render({
  170. treeData: [],
  171. defaultOpen: true,
  172. });
  173. // await sleep();
  174. let opt = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`);
  175. expect(opt.length).toEqual(1);
  176. expect(opt[0].className.includes(`${BASE_CLASS_PREFIX}-cascader-option-empty`)).toEqual(true);
  177. cascader.setProps({ emptyContent: 'test' });
  178. cascader.update();
  179. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0].textContent).toEqual('test');
  180. // done();
  181. });
  182. it('getPopupContainer', () => {
  183. let cascader = render({
  184. getPopupContainer: getPopupContainer,
  185. defaultOpen: true,
  186. });
  187. // await sleep();
  188. let dom = document.querySelector(`.${BASE_CLASS_PREFIX}-cascader`);
  189. expect(dom.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-lists`).length).toEqual(1);
  190. // done();
  191. });
  192. it('defaultOpen', () => {
  193. let cascader = render({
  194. defaultOpen: true,
  195. });
  196. // await sleep();
  197. expect(cascader.state().isOpen).toEqual(true);
  198. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-lists`).length).toEqual(1);
  199. // done();
  200. });
  201. it('custom dropdownClassNam/dropdownStyle/zIndex', () => {
  202. let props = {
  203. dropdownClassName: 'test',
  204. dropdownStyle: {
  205. color: 'red',
  206. },
  207. defaultOpen: true,
  208. zIndex: 2000,
  209. };
  210. let cascader = render(props);
  211. // await sleep();
  212. const dropdown = document.querySelector(`.${BASE_CLASS_PREFIX}-cascader-popover`);
  213. expect(document.querySelector(`.${BASE_CLASS_PREFIX}-portal`).style.zIndex).toEqual('2000');
  214. expect(dropdown.className.includes('test')).toEqual(true);
  215. expect(dropdown.style.color).toEqual('red');
  216. // done();
  217. });
  218. it('defaultValue', () => {
  219. let cascader = render({
  220. defaultValue: ['Yazhou', 'Zhongguo', 'Beijing'],
  221. defaultOpen: true,
  222. });
  223. // await sleep();
  224. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  225. '亚洲 / 中国 / 北京'
  226. );
  227. let lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  228. expect(lists.length).toEqual(3);
  229. let activeItems = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-active`);
  230. expect(activeItems.length).toEqual(2);
  231. expect(activeItems[0].textContent).toEqual('亚洲');
  232. expect(activeItems[1].textContent).toEqual('中国');
  233. let selectItem = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-select`);
  234. expect(selectItem.length).toEqual(1);
  235. expect(selectItem[0].textContent).toEqual('北京');
  236. expect(selectItem[0].querySelectorAll(`.${BASE_CLASS_PREFIX}-icon`).length).toEqual(1);
  237. // done();
  238. });
  239. it('displayProp', () => {
  240. let cascader = render({
  241. defaultValue: ['Yazhou', 'Zhongguo', 'Beijing'],
  242. displayProp: 'value',
  243. });
  244. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  245. 'Yazhou / Zhongguo / Beijing'
  246. );
  247. });
  248. it('displayRender', () => {
  249. let spyOnRender = sinon.spy(() => {});
  250. let cascader = render({
  251. defaultValue: ['Yazhou', 'Zhongguo', 'Beijing'],
  252. displayRender: spyOnRender,
  253. });
  254. expect(spyOnRender.calledWithMatch(['亚洲', '中国', '北京'])).toEqual(true);
  255. cascader.unmount();
  256. let cusRender = render({
  257. defaultValue: ['Yazhou', 'Zhongguo', 'Beijing'],
  258. displayRender: list => '已选择:' + list.join(' -> '),
  259. });
  260. expect(cusRender.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  261. '已选择:亚洲 -> 中国 -> 北京'
  262. );
  263. });
  264. it('disabled', () => {
  265. let cascader = render({
  266. disabled: true,
  267. });
  268. expect(
  269. document
  270. .querySelector(`.${BASE_CLASS_PREFIX}-cascader`)
  271. .className.includes(`${BASE_CLASS_PREFIX}-cascader-disabled`)
  272. ).toEqual(true);
  273. // cannot select
  274. let selectBox = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  275. selectBox.simulate('click');
  276. // await sleep();
  277. expect(cascader.state().isOpen).toEqual(false);
  278. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-lists`).length).toEqual(0);
  279. // done();
  280. });
  281. it('disabled + defaultValue', () => {
  282. const cascaderWithSingle = render({
  283. treeData: treeDataWithDisabled,
  284. changeOnSelect: true,
  285. defaultValue:['Yazhou', 'Zhongguo']
  286. });
  287. expect(
  288. cascaderWithSingle
  289. .find(`.${BASE_CLASS_PREFIX}-cascader-selection span`)
  290. .at(0)
  291. .getDOMNode()
  292. .textContent
  293. ).toEqual('亚洲 / 中国');
  294. cascaderWithSingle.unmount();
  295. const cascaderWithSingleFilter = render({
  296. treeData: treeDataWithDisabled,
  297. changeOnSelect: true,
  298. filterTreeNode: true,
  299. defaultValue:['Yazhou', 'Zhongguo']
  300. });
  301. expect(
  302. cascaderWithSingleFilter
  303. .find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper .${BASE_CLASS_PREFIX}-input`)
  304. .getDOMNode()
  305. .getAttribute('value')
  306. ).toEqual('亚洲 / 中国');
  307. cascaderWithSingleFilter.unmount();
  308. const cascaderWithSingleControlled = render({
  309. treeData: treeDataWithDisabled,
  310. changeOnSelect: true,
  311. value: ['Yazhou', 'Zhongguo'],
  312. });
  313. expect(cascaderWithSingleControlled.find(`.${BASE_CLASS_PREFIX}-cascader-selection span`).getDOMNode().textContent).toEqual(
  314. '亚洲 / 中国'
  315. );
  316. cascaderWithSingleControlled.unmount();
  317. const cascaderWithMultiple = render({
  318. treeData: treeDataWithDisabled,
  319. multiple: true,
  320. defaultValue:['Yazhou', 'Zhongguo']
  321. });
  322. expect(
  323. cascaderWithMultiple
  324. .find(`.${BASE_CLASS_PREFIX}-tag .${BASE_CLASS_PREFIX}-tag-content`)
  325. .at(0)
  326. .getDOMNode()
  327. .textContent
  328. ).toEqual('中国');
  329. cascaderWithMultiple.unmount();
  330. const cascaderWithMultipleFilter = render({
  331. treeData: treeDataWithDisabled,
  332. multiple: true,
  333. filterTreeNode: true,
  334. defaultValue:['Yazhou', 'Zhongguo']
  335. });
  336. expect(
  337. cascaderWithMultipleFilter
  338. .find(`.${BASE_CLASS_PREFIX}-tag .${BASE_CLASS_PREFIX}-tag-content`)
  339. .at(0)
  340. .getDOMNode()
  341. .textContent
  342. ).toEqual('中国');
  343. cascaderWithMultipleFilter.unmount();
  344. const cascaderWithMultipleControlled = render({
  345. treeData: treeDataWithDisabled,
  346. multiple: true,
  347. value:['Yazhou', 'Zhongguo']
  348. });
  349. expect(
  350. cascaderWithMultipleControlled
  351. .find(`.${BASE_CLASS_PREFIX}-tag .${BASE_CLASS_PREFIX}-tag-content`)
  352. .at(0)
  353. .getDOMNode()
  354. .textContent
  355. ).toEqual('中国');
  356. cascaderWithMultipleControlled.unmount();
  357. });
  358. it('select item / onSelect / onChange', () => {
  359. let spyOnSelect = sinon.spy(() => {});
  360. let spyOnChange = sinon.spy(() => {});
  361. let cascader = render({
  362. defaultOpen: true,
  363. onSelect: spyOnSelect,
  364. onChange: spyOnChange,
  365. });
  366. // let selectBox = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  367. // selectBox.simulate('click');
  368. // await sleep();
  369. let lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  370. expect(lists.length).toEqual(1);
  371. lists[0].querySelectorAll('li')[0].click();
  372. lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  373. expect(lists.length).toEqual(2);
  374. expect(spyOnSelect.notCalled).toBe(true);
  375. expect(spyOnChange.notCalled).toBe(true);
  376. expect(
  377. lists[0]
  378. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  379. .className.includes(`${BASE_CLASS_PREFIX}-cascader-option-active`)
  380. ).toEqual(true);
  381. lists[1].querySelectorAll('li')[0].click();
  382. lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  383. expect(lists.length).toEqual(3);
  384. expect(spyOnSelect.notCalled).toBe(true);
  385. expect(spyOnChange.notCalled).toBe(true);
  386. expect(
  387. lists[1]
  388. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  389. .className.includes(`${BASE_CLASS_PREFIX}-cascader-option-active`)
  390. ).toEqual(true);
  391. lists[2].querySelectorAll('li')[0].click();
  392. expect(spyOnSelect.calledOnce).toBe(true);
  393. expect(spyOnChange.calledOnce).toBe(true);
  394. expect(spyOnSelect.calledWithMatch('Beijing')).toEqual(true);
  395. expect(spyOnChange.calledWithMatch(['Yazhou', 'Zhongguo', 'Beijing'])).toEqual(true);
  396. expect(
  397. lists[2]
  398. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  399. .querySelectorAll(`.${BASE_CLASS_PREFIX}-icon-tick`).length
  400. ).toEqual(1);
  401. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  402. '亚洲 / 中国 / 北京'
  403. );
  404. // done();
  405. });
  406. it('filterTreeNode = true shows input box', () => {
  407. let cascader = render({
  408. filterTreeNode: true,
  409. searchPlaceholder: 'placeholder',
  410. });
  411. let searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  412. expect(searchWrapper.exists()).toEqual(true);
  413. expect(
  414. searchWrapper
  415. .find('input')
  416. .instance()
  417. .getAttribute('placeholder')
  418. ).toEqual('placeholder');
  419. });
  420. it('onSearch', () => {
  421. let onSearch = value => {};
  422. let spyOnSearch = sinon.spy(onSearch);
  423. let cascader = render({
  424. filterTreeNode: true,
  425. onSearch: spyOnSearch,
  426. });
  427. const searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  428. let searchValue = '${BASE_CLASS_PREFIX}';
  429. let event = { target: { value: searchValue } };
  430. searchWrapper.find('input').simulate('change', event);
  431. expect(spyOnSearch.calledOnce).toBe(true);
  432. expect(spyOnSearch.calledWithMatch(searchValue)).toBe(true);
  433. });
  434. it('onSelect + changeOnSelect', async () => {
  435. let spyOnSelect = sinon.spy(() => {});
  436. let spyOnChange = sinon.spy(() => {});
  437. let cascader = render({
  438. defaultOpen: true,
  439. onSelect: spyOnSelect,
  440. onChange: spyOnChange,
  441. changeOnSelect: true,
  442. });
  443. // await sleep();
  444. let lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  445. lists[0].querySelectorAll('li')[0].click();
  446. expect(spyOnSelect.calledOnce).toBe(true);
  447. expect(spyOnChange.calledOnce).toBe(true);
  448. expect(spyOnSelect.calledWithMatch('Yazhou')).toEqual(true);
  449. expect(spyOnChange.calledWithMatch(['Yazhou'])).toEqual(true);
  450. expect(
  451. lists[0]
  452. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  453. .className.includes(`${BASE_CLASS_PREFIX}-cascader-option-select`)
  454. ).toEqual(true);
  455. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual('亚洲');
  456. lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  457. lists[1].querySelectorAll('li')[0].click();
  458. expect(spyOnSelect.calledWithMatch('Zhongguo')).toEqual(true);
  459. expect(spyOnChange.calledWithMatch(['Yazhou', 'Zhongguo'])).toEqual(true);
  460. expect(
  461. lists[0]
  462. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  463. .className.includes(`${BASE_CLASS_PREFIX}-cascader-option-active`)
  464. ).toEqual(true);
  465. expect(
  466. lists[1]
  467. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  468. .className.includes(`${BASE_CLASS_PREFIX}-cascader-option-select`)
  469. ).toEqual(true);
  470. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  471. '亚洲 / 中国'
  472. );
  473. });
  474. // display none not working properly
  475. it('filterTreeNode shows correct result', () => {
  476. let cascader = render({
  477. filterTreeNode: true,
  478. defaultOpen: true,
  479. });
  480. // let selectBox = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  481. // selectBox.simulate('click');
  482. // await sleep();
  483. const searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  484. let searchValue = '北';
  485. let event = { target: { value: searchValue } };
  486. searchWrapper.find('input').simulate('change', event);
  487. // await sleep();
  488. let resList = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-label-highlight`);
  489. expect(resList.length).toEqual(3);
  490. // debugger
  491. // expect(resList[0].textContent).toEqual('亚洲/中国/北京');
  492. // done();
  493. });
  494. // display none not working properly
  495. it('filterTreeNode + treeNodeFilterProp', () => {
  496. let cascader = render({
  497. filterTreeNode: true,
  498. treeNodeFilterProp: 'value',
  499. defaultOpen: true,
  500. });
  501. // let selectBox = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  502. // selectBox.simulate('click');
  503. // await sleep();
  504. const searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  505. let searchValue = 'Bei';
  506. let event = { target: { value: searchValue } };
  507. searchWrapper.find('input').simulate('change', event);
  508. // await sleep();
  509. let resList = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-label-highlight`);
  510. expect(resList.length).toEqual(3);
  511. // debugger
  512. // expect(resList[0].textContent).toEqual('亚洲/中国/北京');
  513. // done();
  514. });
  515. it('filterTreeNode + no result', () => {
  516. let cascader = render({
  517. filterTreeNode: true,
  518. defaultOpen: true,
  519. emptyContent: 'test',
  520. });
  521. // await sleep();
  522. const searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  523. let searchValue = 'Bei';
  524. let event = { target: { value: searchValue } };
  525. searchWrapper.find('input').simulate('change', event);
  526. // await sleep();
  527. let opt = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`);
  528. expect(opt.length).toEqual(1);
  529. expect(opt[0].className.includes(`${BASE_CLASS_PREFIX}-cascader-option-empty`)).toEqual(true);
  530. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0].textContent).toEqual('test');
  531. // done();
  532. });
  533. it('filterTreeNode as a func', () => {
  534. let cascader = render({
  535. filterTreeNode: (inputValue, cascaderNode) => cascaderNode === inputValue,
  536. defaultOpen: true,
  537. });
  538. const searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  539. searchWrapper.find('input').simulate('change', { target: { value: '北' } });
  540. // await sleep();
  541. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-option-label-highlight`).length).toEqual(0);
  542. // done();
  543. });
  544. it('controlled: value shows correct', () => {
  545. let cascader = render({
  546. value: ['Yazhou', 'Zhongguo', 'Beijing'],
  547. });
  548. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  549. '亚洲 / 中国 / 北京'
  550. );
  551. });
  552. it('controlled: fire onChange and ui not update', () => {
  553. let spyOnChange = sinon.spy(() => {});
  554. let cascader = render({
  555. value: ['Yazhou', 'Zhongguo', 'Beijing'],
  556. defaultOpen: true,
  557. onChange: spyOnChange,
  558. changeOnSelect: true,
  559. });
  560. // await sleep();
  561. let lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  562. lists[0].querySelectorAll('li')[0].click();
  563. expect(spyOnChange.calledOnce).toBe(true);
  564. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  565. '亚洲 / 中国 / 北京'
  566. );
  567. // done();
  568. });
  569. it('async load data', () => {
  570. const nativeEvent = { nativeEvent: { stopImmediatePropagation: () => {} } };
  571. const then = jest.fn(() => Promise.resolve());
  572. const loadData = jest.fn(() => ({ then }));
  573. const data = {
  574. label: '亚洲',
  575. value: 'Asia',
  576. };
  577. let cascader = render({
  578. loadData,
  579. defaultOpen: true,
  580. treeData: [data],
  581. });
  582. let topNodeAsia = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-option`).at(0);
  583. topNodeAsia.simulate('click', nativeEvent);
  584. expect(loadData).toHaveBeenCalledWith([data]);
  585. expect(then).toHaveBeenCalled();
  586. });
  587. it('select item + onChange + onChangeWithObject', () => {
  588. let spyOnChange = sinon.spy(() => {});
  589. let cascader = render({
  590. defaultOpen: true,
  591. onChange: spyOnChange,
  592. onChangeWithObject: true,
  593. });
  594. // let selectBox = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  595. // selectBox.simulate('click');
  596. // await sleep();
  597. let lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  598. lists[0].querySelectorAll('li')[0].click();
  599. lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  600. lists[1].querySelectorAll('li')[0].click();
  601. lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  602. lists[2].querySelectorAll('li')[0].click();
  603. expect(spyOnChange.calledOnce).toBe(true);
  604. expect(
  605. spyOnChange.calledWithMatch([
  606. {
  607. label: '亚洲',
  608. value: 'Yazhou',
  609. key: 'yazhou',
  610. children: [
  611. {
  612. label: '中国',
  613. value: 'Zhongguo',
  614. key: 'zhongguo',
  615. children: [
  616. {
  617. label: '北京',
  618. value: 'Beijing',
  619. key: 'beijing',
  620. },
  621. {
  622. label: '上海',
  623. value: 'Shanghai',
  624. key: 'shanghai',
  625. },
  626. ],
  627. },
  628. ],
  629. },
  630. {
  631. label: '中国',
  632. value: 'Zhongguo',
  633. key: 'zhongguo',
  634. children: [
  635. {
  636. label: '北京',
  637. value: 'Beijing',
  638. key: 'beijing',
  639. },
  640. {
  641. label: '上海',
  642. value: 'Shanghai',
  643. key: 'shanghai',
  644. },
  645. ],
  646. },
  647. {
  648. label: '北京',
  649. value: 'Beijing',
  650. key: 'beijing',
  651. },
  652. ])
  653. ).toEqual(true);
  654. expect(
  655. lists[2]
  656. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  657. .querySelectorAll(`.${BASE_CLASS_PREFIX}-icon-tick`).length
  658. ).toEqual(1);
  659. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  660. '亚洲 / 中国 / 北京'
  661. );
  662. // done();
  663. });
  664. it('select item + onChange + onChangeWithObject + changeOnSelect', () => {
  665. let spyOnChange = sinon.spy(() => {});
  666. let cascader = render({
  667. defaultOpen: true,
  668. onChange: spyOnChange,
  669. onChangeWithObject: true,
  670. changeOnSelect: true,
  671. });
  672. let lists = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`);
  673. lists[0].querySelectorAll('li')[0].click();
  674. expect(spyOnChange.calledOnce).toBe(true);
  675. expect(
  676. spyOnChange.calledWithMatch([
  677. {
  678. label: '亚洲',
  679. value: 'Yazhou',
  680. key: 'yazhou',
  681. children: [
  682. {
  683. label: '中国',
  684. value: 'Zhongguo',
  685. key: 'zhongguo',
  686. children: [
  687. {
  688. label: '北京',
  689. value: 'Beijing',
  690. key: 'beijing',
  691. },
  692. {
  693. label: '上海',
  694. value: 'Shanghai',
  695. key: 'shanghai',
  696. },
  697. ],
  698. },
  699. ],
  700. },
  701. ])
  702. ).toEqual(true);
  703. expect(
  704. lists[0]
  705. .querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option`)[0]
  706. .className.includes(`${BASE_CLASS_PREFIX}-cascader-option-select`)
  707. ).toEqual(true);
  708. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual('亚洲');
  709. });
  710. it('controlled: value shows correct', () => {
  711. let cascader = render({
  712. value: [
  713. {
  714. label: '亚洲',
  715. value: 'Yazhou',
  716. key: 'yazhou',
  717. children: [
  718. {
  719. label: '中国',
  720. value: 'Zhongguo',
  721. key: 'zhongguo',
  722. children: [
  723. {
  724. label: '北京',
  725. value: 'Beijing',
  726. key: 'beijing',
  727. },
  728. {
  729. label: '上海',
  730. value: 'Shanghai',
  731. key: 'shanghai',
  732. },
  733. ],
  734. },
  735. ],
  736. },
  737. {
  738. label: '中国',
  739. value: 'Zhongguo',
  740. key: 'zhongguo',
  741. children: [
  742. {
  743. label: '北京',
  744. value: 'Beijing',
  745. key: 'beijing',
  746. },
  747. {
  748. label: '上海',
  749. value: 'Shanghai',
  750. key: 'shanghai',
  751. },
  752. ],
  753. },
  754. {
  755. label: '北京',
  756. value: 'Beijing',
  757. key: 'beijing',
  758. },
  759. ],
  760. onChangeWithObject: true,
  761. });
  762. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  763. '亚洲 / 中国 / 北京'
  764. );
  765. });
  766. it('onClear and showClear', () => {
  767. const spyOnClear = sinon.spy(() => { });
  768. const cascader = render({
  769. defaultValue: ['Yazhou', 'Zhongguo', 'Beijing'],
  770. showClear: true,
  771. onClear: spyOnClear,
  772. placeholder: "请选择所在地区"
  773. });
  774. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  775. '亚洲 / 中国 / 北京'
  776. );
  777. cascader.simulate('mouseEnter', {}).find(`.${BASE_CLASS_PREFIX}-cascader-clearbtn`).simulate('click');
  778. expect(spyOnClear.calledOnce).toBe(true);
  779. expect(cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).getDOMNode().textContent).toEqual(
  780. '请选择所在地区'
  781. );
  782. cascader.unmount();
  783. });
  784. it('filterLeafOnly = false', () => {
  785. let cascader = render({
  786. filterTreeNode: true,
  787. defaultOpen: true,
  788. filterLeafOnly: false
  789. });
  790. // let selectBox = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  791. // selectBox.simulate('click');
  792. // await sleep();
  793. const searchWrapper = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-search-wrapper`);
  794. let searchValue = '亚';
  795. let event = { target: { value: searchValue } };
  796. searchWrapper.find('input').simulate('change', event);
  797. // await sleep();
  798. let resList = document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-label-highlight`);
  799. expect(resList.length).toEqual(4);
  800. // debugger
  801. // expect(resList[0].textContent).toEqual('亚洲/中国/北京');
  802. // done();
  803. });
  804. it('showNext',()=>{
  805. // expand next menu by hover
  806. // 通过 hover 展开 Dropdown 的子菜单
  807. const cascaderWithHover = render({
  808. showNext: 'hover',
  809. });
  810. const selectBox = cascaderWithHover.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  811. selectBox.simulate('click');
  812. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`).length).toEqual(1);
  813. cascaderWithHover
  814. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  815. .at(0)
  816. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  817. .at(0)
  818. .simulate('mouseEnter');
  819. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`).length).toEqual(2);
  820. cascaderWithHover.unmount();
  821. // expand next menu by click
  822. // 通过 click 展开 Dropdown 的子菜单
  823. const cascaderWithClick = render({
  824. showNext: 'click',
  825. });
  826. const selectBox2 = cascaderWithClick.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  827. selectBox2.simulate('click');
  828. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`).length).toEqual(1);
  829. cascaderWithClick
  830. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  831. .at(0)
  832. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  833. .at(0)
  834. .simulate('mouseEnter');
  835. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`).length).toEqual(1);
  836. cascaderWithClick
  837. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  838. .at(0)
  839. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  840. .at(0)
  841. .simulate('click');
  842. expect(document.querySelectorAll(`.${BASE_CLASS_PREFIX}-cascader-option-list`).length).toEqual(2);
  843. cascaderWithClick.unmount();
  844. })
  845. it('disableStrictly', ()=>{
  846. // disable strictly
  847. const cascaderWithDisableStrictly = render({
  848. showNext: 'hover',
  849. treeData:treeDataWithDisabled,
  850. disableStrictly: true,
  851. multiple: true,
  852. });
  853. expect(cascaderWithDisableStrictly.state().checkedKeys.size).toEqual(0);
  854. expect(cascaderWithDisableStrictly.state().disabledKeys.size).toEqual(3);
  855. // open dropdown
  856. const selectBoxWithDisableStrictly = cascaderWithDisableStrictly.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  857. selectBoxWithDisableStrictly.simulate('click');
  858. // click checkbox
  859. cascaderWithDisableStrictly
  860. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  861. .at(0)
  862. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  863. .at(0)
  864. .find(`.${BASE_CLASS_PREFIX}-cascader-option-label`)
  865. .at(0)
  866. .find(`.${BASE_CLASS_PREFIX}-checkbox`)
  867. .at(0)
  868. .simulate('click');
  869. // check checkedKeys
  870. expect(cascaderWithDisableStrictly.state().checkedKeys.size).toEqual(1);
  871. cascaderWithDisableStrictly.unmount();
  872. // default disable
  873. const cascaderWithDisable = render({
  874. showNext: 'hover',
  875. treeData:treeDataWithDisabled,
  876. multiple: true,
  877. });
  878. expect(cascaderWithDisable.state().checkedKeys.size).toEqual(0);
  879. expect(cascaderWithDisable.state().disabledKeys.size).toEqual(0);
  880. // open dropdown
  881. const selectBoxWithDisable = cascaderWithDisable.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  882. selectBoxWithDisable.simulate('click');
  883. // click checkbox
  884. cascaderWithDisable
  885. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  886. .at(0)
  887. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  888. .at(0)
  889. .find(`.${BASE_CLASS_PREFIX}-cascader-option-label`)
  890. .at(0)
  891. .find(`.${BASE_CLASS_PREFIX}-checkbox`)
  892. .at(0)
  893. .simulate('click');
  894. // check checkedKeys
  895. expect(cascaderWithDisable.state().checkedKeys.size).toEqual(5);
  896. cascaderWithDisable.unmount();
  897. });
  898. it('multiple + onChangeWithObject', () => {
  899. const cascader = render({
  900. multiple: true,
  901. onChangeWithObject: true,
  902. defaultValue: [
  903. {
  904. label: '北美洲',
  905. value: 'Beimeizhou',
  906. key: 'beimeizhou',
  907. children: [
  908. {
  909. label: '美国',
  910. value: 'Meiguo',
  911. key: 'meiguo',
  912. },
  913. {
  914. label: '加拿大',
  915. value: 'Jianada',
  916. key: 'jianada',
  917. },
  918. ],
  919. },
  920. {
  921. label: '美国',
  922. value: 'Meiguo',
  923. key: 'meiguo',
  924. }
  925. ]
  926. });
  927. const tags = cascader.find(`.${BASE_CLASS_PREFIX}-cascader-selection .${BASE_CLASS_PREFIX}-tag`)
  928. expect(tags.length).toEqual(1);
  929. expect(
  930. tags
  931. .find(`.${BASE_CLASS_PREFIX}-tag-content`)
  932. .getDOMNode()
  933. .textContent
  934. ).toEqual('美国');
  935. });
  936. it('multiple select enableLeafClick', () => {
  937. const cascaderWithMultiple = render({
  938. multiple: true,
  939. enableLeafClick: true,
  940. treeData: [
  941. {
  942. label: '北美洲',
  943. value: 'Beimeizhou',
  944. key: 'beimeizhou',
  945. children: [
  946. {
  947. label: '美国',
  948. value: 'Meiguo',
  949. key: 'meiguo',
  950. },
  951. {
  952. label: '加拿大',
  953. value: 'Jianada',
  954. key: 'jianada',
  955. },
  956. ],
  957. },
  958. {
  959. label: '南美洲',
  960. value: 'Nanmeiguo',
  961. key: 'Nanmeiguo',
  962. }
  963. ]
  964. })
  965. const selectBox = cascaderWithMultiple.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  966. selectBox.simulate('click');
  967. // click checkbox
  968. cascaderWithMultiple
  969. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  970. .at(0)
  971. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  972. .at(1)
  973. .find(`.${BASE_CLASS_PREFIX}-cascader-option-label`)
  974. .at(0)
  975. .find(`.${BASE_CLASS_PREFIX}-checkbox`)
  976. .at(0)
  977. .simulate('click');
  978. expect(cascaderWithMultiple.state().checkedKeys.size).toEqual(1);
  979. // click option cancel checked
  980. cascaderWithMultiple
  981. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  982. .at(0)
  983. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  984. .at(1)
  985. .simulate('click')
  986. expect(cascaderWithMultiple.state().checkedKeys.size).toEqual(0);
  987. // click option select
  988. cascaderWithMultiple
  989. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  990. .at(0)
  991. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  992. .at(0)
  993. .simulate('click')
  994. cascaderWithMultiple
  995. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  996. .at(1)
  997. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  998. .at(0)
  999. .simulate('click')
  1000. expect(cascaderWithMultiple.state().checkedKeys.size).toEqual(1);
  1001. const states = cascaderWithMultiple.state()
  1002. cascaderWithMultiple.unmount();
  1003. })
  1004. it('multiple select disable enableLeafClick', () => {
  1005. const cascaderWithMultiple = render({
  1006. multiple: true,
  1007. treeData: [
  1008. {
  1009. label: '北美洲',
  1010. value: 'Beimeizhou',
  1011. key: 'beimeizhou',
  1012. children: [
  1013. {
  1014. label: '美国',
  1015. value: 'Meiguo',
  1016. key: 'meiguo',
  1017. },
  1018. {
  1019. label: '加拿大',
  1020. value: 'Jianada',
  1021. key: 'jianada',
  1022. },
  1023. ],
  1024. },
  1025. {
  1026. label: '南美洲',
  1027. value: 'Nanmeiguo',
  1028. key: 'Nanmeiguo',
  1029. }
  1030. ]
  1031. })
  1032. const selectBox = cascaderWithMultiple.find(`.${BASE_CLASS_PREFIX}-cascader-selection`).at(0);
  1033. selectBox.simulate('click');
  1034. // click checkbox
  1035. cascaderWithMultiple
  1036. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  1037. .at(0)
  1038. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  1039. .at(1)
  1040. .find(`.${BASE_CLASS_PREFIX}-cascader-option-label`)
  1041. .at(0)
  1042. .find(`.${BASE_CLASS_PREFIX}-checkbox`)
  1043. .at(0)
  1044. .simulate('click');
  1045. expect(cascaderWithMultiple.state().checkedKeys.size).toEqual(1);
  1046. // click option can't cancel checked
  1047. cascaderWithMultiple
  1048. .find(`.${BASE_CLASS_PREFIX}-cascader-option-list`)
  1049. .at(0)
  1050. .find(`.${BASE_CLASS_PREFIX}-cascader-option`)
  1051. .at(1)
  1052. .simulate('click')
  1053. expect(cascaderWithMultiple.state().checkedKeys.size).toEqual(1);
  1054. cascaderWithMultiple.unmount();
  1055. })
  1056. it('triggerRender', () => {
  1057. const spyTriggerRender = sinon.spy(() => <span>123</span>);
  1058. const cascaderAutoMerge = render({
  1059. multiple: true,
  1060. triggerRender: spyTriggerRender,
  1061. defaultValue: 'Yazhou'
  1062. });
  1063. cascaderAutoMerge.simulate('click');
  1064. const firstCall = spyTriggerRender.getCall(0);
  1065. const args = firstCall.args[0];
  1066. /* check arguments of triggerRender */
  1067. expect(args.value.size).toEqual(1);
  1068. expect(args.value).toEqual(new Set('0'));
  1069. cascaderAutoMerge.unmount();
  1070. const spyTriggerRender2 = sinon.spy(() => <span>123</span>);
  1071. const cascaderNoAutoMerge = render({
  1072. multiple: true,
  1073. triggerRender: spyTriggerRender2,
  1074. defaultValue: 'Yazhou',
  1075. autoMergeValue: false,
  1076. });
  1077. cascaderNoAutoMerge.simulate('click');
  1078. const firstCall2 = spyTriggerRender2.getCall(0);
  1079. const args2 = firstCall2.args[0];
  1080. /* check arguments of triggerRender */
  1081. expect(args2.value.size).toEqual(4);
  1082. expect(args2.value).toEqual(new Set(['0','0-0','0-0-1','0-0-0']));
  1083. cascaderNoAutoMerge.unmount();
  1084. });
  1085. });