table.spec.js 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413
  1. import { createVue, triggerEvent, destroyVM } from '../util';
  2. const DELAY = 10;
  3. const testDataArr = [];
  4. const toArray = function(obj) {
  5. return [].slice.call(obj);
  6. };
  7. const getTestData = function() {
  8. return [
  9. { name: 'Toy Story', release: '1995-11-22', director: 'John Lasseter', runtime: 80 },
  10. { name: 'A Bug\'s Life', release: '1998-11-25', director: 'John Lasseter', runtime: 95 },
  11. { name: 'Toy Story 2', release: '1999-11-24', director: 'John Lasseter', runtime: 92 },
  12. { name: 'Monsters, Inc.', release: '2001-11-2', director: 'Peter Docter', runtime: 92 },
  13. { name: 'Finding Nemo', release: '2003-5-30', director: 'Andrew Stanton', runtime: 100 }
  14. ];
  15. };
  16. getTestData().forEach(cur => {
  17. Object.keys(cur).forEach(prop => {
  18. testDataArr.push(cur[prop].toString());
  19. });
  20. });
  21. describe('Table', () => {
  22. describe('rendering data is correct', () => {
  23. const vm = createVue({
  24. template: `
  25. <el-table :data="testData">
  26. <el-table-column prop="name" label="片名" />
  27. <el-table-column prop="release" label="发行日期" />
  28. <el-table-column prop="director" label="导演" />
  29. <el-table-column prop="runtime" label="时长(分)" />
  30. </el-table>
  31. `,
  32. created() {
  33. this.testData = getTestData();
  34. }
  35. });
  36. it('head', done => {
  37. setTimeout(_ => {
  38. const ths = toArray(vm.$el.querySelectorAll('thead th'));
  39. expect(ths.map(node => node.textContent).filter(o => o))
  40. .to.eql(['片名', '发行日期', '导演', '时长(分)']);
  41. done();
  42. }, DELAY);
  43. });
  44. it('row length', () => {
  45. expect(vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr')).to.length(getTestData().length);
  46. });
  47. it('row data', () => {
  48. const cells = toArray(vm.$el.querySelectorAll('td .cell'))
  49. .map(node => node.textContent);
  50. expect(cells).to.eql(testDataArr);
  51. destroyVM(vm);
  52. });
  53. });
  54. describe('attributes', () => {
  55. const createTable = function(props, opts) {
  56. return createVue(Object.assign({
  57. template: `
  58. <el-table :data="testData" ${props}>
  59. <el-table-column prop="name" label="片名" />
  60. <el-table-column prop="release" label="发行日期" />
  61. <el-table-column prop="director" label="导演" />
  62. <el-table-column prop="runtime" label="时长(分)" />
  63. </el-table>
  64. `,
  65. created() {
  66. this.testData = getTestData();
  67. }
  68. }, opts));
  69. };
  70. it('height', done => {
  71. const vm = createTable('height="134"');
  72. setTimeout(_ => {
  73. expect(vm.$el.style.height).to.equal('134px');
  74. destroyVM(vm);
  75. done();
  76. }, DELAY);
  77. });
  78. it('stripe', done => {
  79. const vm = createTable('stripe');
  80. setTimeout(_ => {
  81. expect(vm.$el.classList.contains('el-table--striped')).to.true;
  82. destroyVM(vm);
  83. done();
  84. }, DELAY);
  85. });
  86. it('border', done => {
  87. const vm = createTable('border');
  88. setTimeout(_ => {
  89. expect(vm.$el.classList.contains('el-table--border')).to.true;
  90. destroyVM(vm);
  91. done();
  92. }, DELAY);
  93. });
  94. it('fit', done => {
  95. const vm = createTable(':fit="false"');
  96. setTimeout(_ => {
  97. expect(vm.$el.classList.contains('el-table--fit')).to.false;
  98. destroyVM(vm);
  99. done();
  100. }, DELAY);
  101. });
  102. it('show-header', done => {
  103. const vm = createTable(':show-header="false"');
  104. setTimeout(_ => {
  105. expect(vm.$el.querySelectorAll('.el-table__header-wrapper').length).to.equal(0);
  106. destroyVM(vm);
  107. done();
  108. }, DELAY);
  109. });
  110. it('tableRowClassName', done => {
  111. const vm = createTable(':row-class-name="tableRowClassName"', {
  112. methods: {
  113. tableRowClassName(row, index) {
  114. if (index === 1) {
  115. return 'info-row';
  116. } else if (index === 3) {
  117. return 'positive-row';
  118. }
  119. return '';
  120. }
  121. }
  122. });
  123. setTimeout(_ => {
  124. expect(vm.$el.querySelectorAll('.info-row')).to.length(1);
  125. expect(vm.$el.querySelectorAll('.positive-row')).to.length(1);
  126. destroyVM(vm);
  127. done();
  128. }, DELAY);
  129. });
  130. });
  131. describe('filter', () => {
  132. let vm;
  133. beforeEach(done => {
  134. vm = createVue({
  135. template: `
  136. <el-table ref="table" :data="testData">
  137. <el-table-column prop="name" label="片名" />
  138. <el-table-column prop="release" label="发行日期" />
  139. <el-table-column
  140. prop="director"
  141. :filters="[
  142. { text: 'John Lasseter', value: 'John Lasseter' },
  143. { text: 'Peter Docter', value: 'Peter Docter' },
  144. { text: 'Andrew Stanton', value: 'Andrew Stanton' }
  145. ]"
  146. :filter-method="filterMethod"
  147. label="导演" />
  148. <el-table-column prop="runtime" label="时长(分)" />
  149. </el-table>
  150. `,
  151. created() {
  152. this.testData = getTestData();
  153. },
  154. methods: {
  155. filterMethod(value, row) {
  156. return value === row.director;
  157. }
  158. }
  159. }, true);
  160. setTimeout(done, DELAY);
  161. });
  162. afterEach(() => destroyVM(vm));
  163. it('render', () => {
  164. expect(vm.$el.querySelector('.el-table__column-filter-trigger')).to.exist;
  165. });
  166. it('click dropdown', done => {
  167. const btn = vm.$el.querySelector('.el-table__column-filter-trigger');
  168. triggerEvent(btn, 'click', true, false);
  169. setTimeout(_ => {
  170. const filter = document.body.querySelector('.el-table-filter');
  171. expect(filter).to.exist;
  172. document.body.removeChild(filter);
  173. done();
  174. }, 100);
  175. });
  176. it('click filter', done => {
  177. const btn = vm.$el.querySelector('.el-table__column-filter-trigger');
  178. triggerEvent(btn, 'click', true, false);
  179. setTimeout(_ => {
  180. const filter = document.body.querySelector('.el-table-filter');
  181. // John Lasseter
  182. triggerEvent(filter.querySelector('.el-checkbox'), 'click', true, false);
  183. // confrim button
  184. setTimeout(_ => {
  185. triggerEvent(filter.querySelector('.el-table-filter__bottom button'), 'click', true, false);
  186. setTimeout(_ => {
  187. expect(vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr')).to.length(3);
  188. document.body.removeChild(filter);
  189. done();
  190. }, DELAY);
  191. }, 100);
  192. }, 100);
  193. });
  194. it('click reset', done => {
  195. const btn = vm.$el.querySelector('.el-table__column-filter-trigger');
  196. triggerEvent(btn, 'click', true, false);
  197. setTimeout(_ => {
  198. const filter = document.body.querySelector('.el-table-filter');
  199. // John Lasseter
  200. triggerEvent(filter.querySelector('.el-checkbox'), 'click', true, false);
  201. setTimeout(_ => {
  202. // reset button
  203. triggerEvent(filter.querySelectorAll('.el-table-filter__bottom button')[1], 'click', true, false);
  204. setTimeout(_ => {
  205. expect(filter.querySelector('.el-table-filter__bottom button').classList.contains('is-disabled')).to.true;
  206. document.body.removeChild(filter);
  207. destroyVM(vm);
  208. done();
  209. }, DELAY);
  210. }, 100);
  211. }, 100);
  212. });
  213. });
  214. describe('events', () => {
  215. const createTable = function(prop = '', opts) {
  216. return createVue({
  217. template: `
  218. <el-table :data="testData" @${prop}="handleEvent">
  219. <el-table-column type="selection" />
  220. <el-table-column prop="name" />
  221. <el-table-column prop="release" />
  222. <el-table-column prop="director" />
  223. <el-table-column prop="runtime"/>
  224. </el-table>
  225. `,
  226. methods: {
  227. handleEvent(...args) {
  228. this.result = args;
  229. }
  230. },
  231. created() {
  232. this.testData = getTestData();
  233. },
  234. data() {
  235. return { result: '', testData: this.testData };
  236. }
  237. }, true);
  238. };
  239. it('select', done => {
  240. const vm = createTable('select');
  241. setTimeout(_ => {
  242. vm.$el.querySelectorAll('.el-checkbox')[1].click();
  243. expect(vm.result).to.length(2);
  244. expect(vm.result[1]).to.have.property('name').to.equal(getTestData()[0].name);
  245. destroyVM(vm);
  246. done();
  247. }, DELAY);
  248. });
  249. it('select-all', done => {
  250. const vm = createTable('select-all');
  251. setTimeout(_ => {
  252. vm.$el.querySelector('.el-checkbox').click();
  253. setTimeout(_ => {
  254. expect(vm.result).to.length(1);
  255. expect(vm.result[0]).to.length(getTestData().length);
  256. destroyVM(vm);
  257. done();
  258. }, DELAY);
  259. }, DELAY);
  260. });
  261. it('selection-change', done => {
  262. const vm = createTable('selection-change');
  263. setTimeout(_ => {
  264. vm.$el.querySelectorAll('.el-checkbox')[1].click();
  265. expect(vm.result).to.length(1);
  266. destroyVM(vm);
  267. done();
  268. }, DELAY);
  269. });
  270. it('cell-mouse-enter', done => {
  271. const vm = createTable('cell-mouse-enter');
  272. setTimeout(_ => {
  273. const cell = vm.$el.querySelectorAll('.el-table__body .cell')[2]; // first row
  274. triggerEvent(cell.parentNode, 'mouseenter');
  275. expect(vm.result).to.length(4); // row, column, cell, event
  276. expect(vm.result[0]).to.have.property('name').to.equal(getTestData()[0].name);
  277. destroyVM(vm);
  278. done();
  279. }, DELAY);
  280. });
  281. it('cell-mouse-leave', done => {
  282. const vm = createTable('cell-mouse-leave');
  283. setTimeout(_ => {
  284. const cell = vm.$el.querySelectorAll('.el-table__body .cell')[7]; // second row
  285. const cell2 = vm.$el.querySelectorAll('.el-table__body .cell')[2]; // first row
  286. triggerEvent(cell2.parentNode, 'mouseenter');
  287. triggerEvent(cell.parentNode, 'mouseleave');
  288. expect(vm.result).to.length(4); // row, column, cell, event
  289. expect(vm.result[0]).to.have.property('name').to.equal(getTestData()[0].name);
  290. destroyVM(vm);
  291. done();
  292. }, DELAY);
  293. });
  294. it('cell-click', done => {
  295. const vm = createTable('cell-click');
  296. setTimeout(_ => {
  297. const cell = vm.$el.querySelectorAll('.el-table__body .cell')[2]; // first row
  298. cell.parentNode.click();
  299. expect(vm.result).to.length(4); // row, column, cell, event
  300. expect(vm.result[0]).to.have.property('name').to.equal(getTestData()[0].name);
  301. destroyVM(vm);
  302. done();
  303. }, DELAY);
  304. });
  305. it('row-click', done => {
  306. const vm = createTable('row-click');
  307. setTimeout(_ => {
  308. const cell = vm.$el.querySelectorAll('.el-table__body .cell')[2]; // first row
  309. triggerEvent(cell.parentNode.parentNode, 'click');
  310. expect(vm.result).to.length(2); // row, event
  311. expect(vm.result[0]).to.have.property('name').to.equal(getTestData()[0].name);
  312. destroyVM(vm);
  313. done();
  314. }, DELAY);
  315. });
  316. it('current-change', done => {
  317. const vm = createTable('current-change');
  318. setTimeout(_ => {
  319. const cell = vm.$el.querySelectorAll('.el-table__body .cell')[2]; // first row
  320. triggerEvent(cell.parentNode.parentNode, 'click');
  321. expect(vm.result).to.length(2); // currentRow, oldCurrentRow
  322. expect(vm.result[0]).to.have.property('name').to.equal(getTestData()[0].name);
  323. expect(vm.result[1]).to.equal(null);
  324. // clear data => current-change should fire again.
  325. const oldRow = vm.result[0];
  326. vm.testData = [];
  327. setTimeout(() => {
  328. expect(vm.result).to.length(2); // currentRow, oldCurrentRow
  329. expect(vm.result[0]).to.equal(null);
  330. expect(vm.result[1]).to.equal(oldRow);
  331. destroyVM(vm);
  332. done();
  333. }, DELAY);
  334. }, DELAY);
  335. });
  336. it('header-click', done => {
  337. const vm = createTable('header-click');
  338. setTimeout(_ => {
  339. const cell = vm.$el.querySelectorAll('.el-table__header th')[1]; // header[prop='name']
  340. triggerEvent(cell, 'click');
  341. expect(vm.result).to.length(2); // column, event
  342. expect(vm.result[0]).to.have.property('property').to.equal('name');
  343. destroyVM(vm);
  344. done();
  345. }, DELAY);
  346. });
  347. });
  348. describe('column attributes', () => {
  349. const createTable = function(props1, props2, props3, props4, opts, tableProps) {
  350. return createVue(Object.assign({
  351. template: `
  352. <el-table :data="testData" ${tableProps || ''}>
  353. <el-table-column prop="name" ${props1 || ''} />
  354. <el-table-column prop="release" ${props2 || ''} />
  355. <el-table-column prop="director" ${props3 || ''} />
  356. <el-table-column prop="runtime" ${props4 || ''} />
  357. </el-table>
  358. `,
  359. created() {
  360. this.testData = getTestData();
  361. }
  362. }, opts));
  363. };
  364. it('label', done => {
  365. const vm = createTable('label="啊哈哈哈"', 'label="啊啦啦啦"');
  366. setTimeout(_ => {
  367. const ths = toArray(vm.$el.querySelectorAll('thead th'))
  368. .map(node => node.textContent).filter(o => o);
  369. expect(ths).to.eql(['啊哈哈哈', '啊啦啦啦']);
  370. destroyVM(vm);
  371. done();
  372. }, DELAY);
  373. });
  374. it('width', done => {
  375. const vm = createTable('width="123px"', ':width="102"', 'width="39"');
  376. setTimeout(_ => {
  377. const ths = toArray(vm.$el.querySelectorAll('.el-table__header-wrapper col'))
  378. .map(node => node.width).filter(o => o);
  379. expect(ths).to.include('123').include('102').include('39');
  380. destroyVM(vm);
  381. done();
  382. }, DELAY);
  383. });
  384. it('fixed', done => {
  385. const vm = createTable(
  386. 'fixed label="test1"',
  387. 'fixed="right" label="test2"',
  388. 'fixed="left" label="test3"');
  389. setTimeout(_ => {
  390. expect(toArray(vm.$el.querySelectorAll('.el-table__fixed th:not(.is-hidden)'))
  391. .map(node => node.textContent))
  392. .to.eql(['test1', 'test3']);
  393. expect(toArray(vm.$el.querySelectorAll('.el-table__fixed-right th:not(.is-hidden)'))
  394. .map(node => node.textContent))
  395. .to.eql(['test2']);
  396. expect(vm.$el.querySelector('.el-table__body-wrapper').style.height).to.equal('');
  397. destroyVM(vm);
  398. done();
  399. }, DELAY);
  400. });
  401. it('resizable', done => {
  402. const vm = createTable(
  403. 'resizable',
  404. ':resizable="false"',
  405. '',
  406. '',
  407. {},
  408. 'border');
  409. setTimeout(_ => {
  410. const firstCol = vm.$el.querySelector('thead th');
  411. triggerEvent(firstCol, 'mousemove');
  412. triggerEvent(firstCol, 'mousedown');
  413. destroyVM(vm);
  414. done();
  415. }, DELAY);
  416. });
  417. it('formatter', done => {
  418. const vm = createTable(
  419. ':formatter="renderCell"', '', '', '', {
  420. methods: {
  421. renderCell(row, column) {
  422. return `[${row.name}]`;
  423. }
  424. }
  425. });
  426. setTimeout(_ => {
  427. const cells = toArray(vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr td:first-child'));
  428. expect(cells.map(n => n.textContent)).to.eql(getTestData().map(o => `[${o.name}]`));
  429. destroyVM(vm);
  430. done();
  431. }, DELAY);
  432. });
  433. it('show-overflow-tooltip', done => {
  434. const vm = createTable('show-overflow-tooltip');
  435. setTimeout(_ => {
  436. expect(vm.$el.querySelectorAll('.el-tooltip')).to.length(5);
  437. destroyVM(vm);
  438. done();
  439. }, DELAY);
  440. });
  441. it('show-tooltip-when-overflow', done => { // old version prop name
  442. const vm = createTable('show-tooltip-when-overflow');
  443. setTimeout(_ => {
  444. expect(vm.$el.querySelectorAll('.el-tooltip')).to.length(5);
  445. destroyVM(vm);
  446. done();
  447. }, DELAY);
  448. });
  449. it('inline-template', done => {
  450. const vm = createVue({
  451. template: `
  452. <el-table :data="testData">
  453. <el-table-column prop="name" inline-template>
  454. <span>[{{ row.name }}]</span>
  455. </el-table-column>
  456. <el-table-column prop="release"/>
  457. <el-table-column prop="director"/>
  458. <el-table-column prop="runtime"/>
  459. </el-table>
  460. `,
  461. created() {
  462. this.testData = getTestData();
  463. }
  464. });
  465. setTimeout(_ => {
  466. const cells = toArray(vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr td:first-child'));
  467. expect(cells.map(n => n.textContent)).to.eql(getTestData().map(o => `[${o.name}]`));
  468. destroyVM(vm);
  469. done();
  470. }, DELAY);
  471. });
  472. it('render-header', done => {
  473. const vm = createVue({
  474. template: `
  475. <el-table :data="testData">
  476. <el-table-column prop="name" :render-header="renderHeader" label="name">
  477. </el-table-column>
  478. <el-table-column prop="release"/>
  479. <el-table-column prop="director"/>
  480. <el-table-column prop="runtime"/>
  481. </el-table>
  482. `,
  483. methods: {
  484. renderHeader(h, { column, $index }) {
  485. return '' + $index + ':' + column.label;
  486. }
  487. },
  488. created() {
  489. this.testData = getTestData();
  490. }
  491. });
  492. setTimeout(_ => {
  493. const headerCell = vm.$el.querySelector('.el-table__header-wrapper thead tr th:first-child .cell');
  494. expect(headerCell.textContent).to.equal('0:name');
  495. destroyVM(vm);
  496. done();
  497. }, DELAY);
  498. });
  499. it('align', done => {
  500. const vm = createTable('align="left"', 'align="right"', 'align="center"');
  501. setTimeout(_ => {
  502. var len = getTestData().length + 1;
  503. expect(vm.$el.querySelectorAll('.is-left')).to.length(len);
  504. expect(vm.$el.querySelectorAll('.is-right')).to.length(len);
  505. expect(vm.$el.querySelectorAll('.is-center')).to.length(len);
  506. destroyVM(vm);
  507. done();
  508. }, DELAY);
  509. });
  510. it('class-name', done => {
  511. const vm = createTable('class-name="column-1"', 'class-name="column-2 column-class-a"', 'class-name="column-class-a"');
  512. setTimeout(_ => {
  513. var len = getTestData().length + 1;
  514. expect(vm.$el.querySelectorAll('.column-1')).to.length(len);
  515. expect(vm.$el.querySelectorAll('.column-2')).to.length(len);
  516. expect(vm.$el.querySelectorAll('.column-class-a')).to.length(len * 2);
  517. destroyVM(vm);
  518. done();
  519. }, DELAY);
  520. });
  521. it('selectable', done => {
  522. const vm = createVue({
  523. template: `
  524. <el-table :data="testData" @selection-change="change">
  525. <el-table-column type="selection" :selectable="filterSelect" />
  526. <el-table-column prop="name" label="name" />
  527. <el-table-column prop="release" label="release" />
  528. <el-table-column prop="director" label="director" />
  529. <el-table-column prop="runtime" label="runtime" />
  530. </el-table>
  531. `,
  532. created() {
  533. this.testData = getTestData();
  534. },
  535. data() {
  536. return { selected: [] };
  537. },
  538. methods: {
  539. change(rows) {
  540. this.selected = rows;
  541. },
  542. filterSelect(row, index) {
  543. return index > 2;
  544. }
  545. }
  546. }, true);
  547. setTimeout(_ => {
  548. vm.$el.querySelector('.el-checkbox').click();
  549. setTimeout(_ => {
  550. expect(vm.selected).to.length(2);
  551. destroyVM(vm);
  552. done();
  553. }, DELAY);
  554. }, DELAY);
  555. });
  556. it('selectable === false & check selectAll status', done => {
  557. const vm = createVue({
  558. template: `
  559. <el-table :data="testData" @selection-change="change">
  560. <el-table-column type="selection" :selectable="filterSelect" />
  561. <el-table-column prop="name" label="name" />
  562. <el-table-column prop="release" label="release" />
  563. <el-table-column prop="director" label="director" />
  564. <el-table-column prop="runtime" label="runtime" />
  565. </el-table>
  566. `,
  567. created() {
  568. },
  569. data() {
  570. return { selected: [], testData: null };
  571. },
  572. methods: {
  573. change(rows) {
  574. this.selected = rows;
  575. },
  576. filterSelect(row, index) {
  577. return false;
  578. }
  579. }
  580. }, true);
  581. vm.testData = getTestData();
  582. setTimeout(_ => {
  583. expect(vm.$el.querySelector('.el-checkbox').__vue__.checked).to.be.false;
  584. setTimeout(_ => {
  585. expect(vm.selected).to.length(0);
  586. destroyVM(vm);
  587. done();
  588. }, DELAY);
  589. }, DELAY);
  590. });
  591. it('emit selection-change after row has been removed', done => {
  592. const vm = createVue({
  593. template: `
  594. <el-table :data="testData" @selection-change="change">
  595. <el-table-column type="selection" />
  596. <el-table-column prop="name" label="name" />
  597. <el-table-column prop="release" label="release" />
  598. <el-table-column prop="director" label="director" />
  599. <el-table-column prop="runtime" label="runtime" />
  600. </el-table>
  601. `,
  602. created() {
  603. this.testData = getTestData();
  604. },
  605. data() {
  606. return { selected: [], testData: null };
  607. },
  608. methods: {
  609. change(rows) {
  610. this.selected = rows;
  611. },
  612. filterSelect(row, index) {
  613. return index > 2;
  614. }
  615. }
  616. }, true);
  617. setTimeout(_ => {
  618. vm.$el.querySelector('.el-checkbox').click();
  619. setTimeout(_ => {
  620. expect(vm.selected).to.length(5);
  621. vm.testData.splice(0, 1);
  622. setTimeout(_ => {
  623. expect(vm.selected).to.length(4);
  624. destroyVM(vm);
  625. done();
  626. });
  627. }, DELAY);
  628. }, DELAY);
  629. });
  630. it('reserve-selection', done => {
  631. const getData = function(page = 0) {
  632. let id = 0;
  633. const rows = [];
  634. const row = () => {
  635. return {
  636. id: ++id + page * 10,
  637. date: new Date().getTime()
  638. };
  639. };
  640. let count = 10;
  641. while (--count) {
  642. rows.push(row());
  643. }
  644. return rows;
  645. };
  646. const vm = createVue({
  647. template: `
  648. <el-table ref="table" :row-key="rowKey" :data="testData" @selection-change="change">
  649. <el-table-column type="selection" reserve-selection />
  650. <el-table-column prop="id" label="id" />
  651. <el-table-column prop="date" label="date" />
  652. </el-table>
  653. `,
  654. created() {
  655. this.testData = getData();
  656. },
  657. data() {
  658. return { selected: [], testData: [] };
  659. },
  660. methods: {
  661. rowKey(row) {
  662. return row.id;
  663. },
  664. change(rows) {
  665. this.selected = rows;
  666. }
  667. }
  668. }, true);
  669. setTimeout(_ => {
  670. // click first
  671. vm.$el.querySelectorAll('.el-checkbox')[1].click();
  672. setTimeout(_ => {
  673. expect(vm.$el.querySelectorAll('.el-checkbox__inner.is-checked')).to.length(1);
  674. // go to second page
  675. vm.testData = getData(1);
  676. setTimeout(_ => {
  677. // expect no checked
  678. expect(vm.$el.querySelectorAll('.el-checkbox__inner.is-checked')).to.length(0);
  679. // click first checkbox
  680. vm.$el.querySelectorAll('.el-checkbox')[1].click();
  681. vm.$el.querySelectorAll('.el-checkbox')[2].click();
  682. setTimeout(_ => {
  683. // back first page
  684. vm.testData = getData();
  685. setTimeout(_ => {
  686. expect(vm.$el.querySelectorAll('.el-checkbox__inner.is-checked')).to.length(1);
  687. // clear
  688. vm.$refs.table.clearSelection();
  689. setTimeout(_ => {
  690. expect(vm.$el.querySelectorAll('.el-checkbox__inner.is-checked')).to.length(0);
  691. destroyVM(vm);
  692. done();
  693. }, DELAY);
  694. }, DELAY);
  695. }, DELAY);
  696. }, DELAY);
  697. }, DELAY);
  698. }, DELAY);
  699. });
  700. describe('type', () => {
  701. const createTable = function(type) {
  702. return createVue({
  703. template: `
  704. <el-table :data="testData" @selection-change="change">
  705. <el-table-column type="${type}" />
  706. <el-table-column prop="name" label="name" />
  707. <el-table-column prop="release" label="release" />
  708. <el-table-column prop="director" label="director" />
  709. <el-table-column prop="runtime" label="runtime" />
  710. </el-table>
  711. `,
  712. created() {
  713. this.testData = getTestData();
  714. },
  715. data() {
  716. return { selected: [] };
  717. },
  718. methods: {
  719. change(rows) {
  720. this.selected = rows;
  721. }
  722. }
  723. }, true);
  724. };
  725. describe('= selection', () => {
  726. const vm = createTable('selection');
  727. it('render', done => {
  728. setTimeout(_ => {
  729. expect(vm.$el.querySelectorAll('.el-checkbox')).to.length(getTestData().length + 1);
  730. done();
  731. }, DELAY);
  732. });
  733. it('select all', done => {
  734. vm.$el.querySelector('.el-checkbox').click();
  735. setTimeout(_ => {
  736. expect(vm.selected).to.length(getTestData().length);
  737. done();
  738. }, DELAY);
  739. });
  740. it('cancel all', done => {
  741. vm.$el.querySelector('.el-checkbox').click();
  742. setTimeout(_ => {
  743. expect(vm.selected).to.length(0);
  744. destroyVM(vm);
  745. done();
  746. }, DELAY);
  747. });
  748. it('select one', done => {
  749. const vm2 = createTable('selection');
  750. setTimeout(_ => {
  751. vm2.$el.querySelectorAll('.el-checkbox')[1].click();
  752. setTimeout(_ => {
  753. expect(vm2.selected).to.length(1);
  754. expect(vm2.selected[0].name).to.equal(getTestData()[0].name);
  755. destroyVM(vm2);
  756. done();
  757. }, DELAY);
  758. }, DELAY);
  759. });
  760. });
  761. describe('= index', () => {
  762. const vm = createTable('index');
  763. it('render', done => {
  764. setTimeout(_ => {
  765. expect(toArray(vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr td:first-child'))
  766. .map(node => node.textContent)).to.eql(['1', '2', '3', '4', '5']);
  767. destroyVM(vm);
  768. done();
  769. }, DELAY);
  770. });
  771. });
  772. });
  773. describe('sortable', () => {
  774. it('render', done => {
  775. const vm = createTable('', '', '', 'sortable');
  776. setTimeout(_ => {
  777. expect(vm.$el.querySelectorAll('.caret-wrapper')).to.length(1);
  778. destroyVM(vm);
  779. done();
  780. }, DELAY);
  781. });
  782. it('sortable method', done => {
  783. const vm = createTable(
  784. 'sortable :sort-method="sortMethod"', '', '', '', {
  785. methods: {
  786. sortMethod(a, b) {
  787. return a.runtime < b.runtime;
  788. }
  789. }
  790. });
  791. setTimeout(_ => {
  792. const elm = vm.$el.querySelector('.caret-wrapper');
  793. elm.click();
  794. setTimeout(_ => {
  795. const lastCells = vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr td:last-child');
  796. expect(toArray(lastCells).map(node => node.textContent)).to.eql(['100', '95', '92', '92', '80']);
  797. destroyVM(vm);
  798. done();
  799. }, DELAY);
  800. }, DELAY);
  801. });
  802. it('sort-change', done => {
  803. let result;
  804. const vm = createTable('sortable="custom"', '', '', '', {
  805. methods: {
  806. sortChange(...args) {
  807. result = args;
  808. }
  809. }
  810. }, '@sort-change="sortChange"');
  811. setTimeout(_ => {
  812. const elm = vm.$el.querySelector('.caret-wrapper');
  813. elm.click();
  814. setTimeout(_ => {
  815. expect(result).to.exist;
  816. destroyVM(vm);
  817. done();
  818. }, DELAY);
  819. }, DELAY);
  820. });
  821. });
  822. describe('click sortable column', () => {
  823. const vm = createTable('', '', '', 'sortable');
  824. it('ascending', done => {
  825. const elm = vm.$el.querySelector('.caret-wrapper');
  826. elm.click();
  827. setTimeout(_ => {
  828. const lastCells = vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr td:last-child');
  829. expect(toArray(lastCells).map(node => node.textContent))
  830. .to.eql(['80', '92', '92', '95', '100']);
  831. done();
  832. }, DELAY);
  833. });
  834. it('descending', done => {
  835. const elm = vm.$el.querySelector('.caret-wrapper');
  836. elm.click();
  837. setTimeout(_ => {
  838. const lastCells = vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr td:last-child');
  839. expect(toArray(lastCells).map(node => node.textContent))
  840. .to.eql(['100', '95', '92', '92', '80']);
  841. destroyVM(vm);
  842. done();
  843. }, DELAY);
  844. });
  845. });
  846. });
  847. describe('multi level column', () => {
  848. it('should works', done => {
  849. const vm = createVue({
  850. template: `
  851. <el-table :data="testData">
  852. <el-table-column prop="name" />
  853. <el-table-column label="group">
  854. <el-table-column prop="release"/>
  855. <el-table-column prop="director"/>
  856. </el-table-column>
  857. <el-table-column prop="runtime"/>
  858. </el-table>
  859. `,
  860. created() {
  861. this.testData = null;
  862. }
  863. }, true);
  864. setTimeout(_ => {
  865. const trs = vm.$el.querySelectorAll('.el-table__header tr');
  866. expect(trs.length).equal(2);
  867. const firstRowHeader = trs[0].querySelectorAll('th .cell').length;
  868. const secondRowHeader = trs[1].querySelectorAll('th .cell').length;
  869. expect(firstRowHeader).to.equal(3);
  870. expect(secondRowHeader).to.equal(2);
  871. expect(trs[0].querySelector('th:first-child').getAttribute('rowspan')).to.equal('2');
  872. expect(trs[0].querySelector('th:nth-child(2)').getAttribute('colspan')).to.equal('2');
  873. destroyVM(vm);
  874. done();
  875. }, DELAY);
  876. });
  877. it('should works', done => {
  878. const vm = createVue({
  879. template: `
  880. <el-table :data="testData">
  881. <el-table-column prop="name" />
  882. <el-table-column label="group">
  883. <el-table-column label="group's group">
  884. <el-table-column prop="release" />
  885. <el-table-column prop="runtime"/>
  886. </el-table-column>
  887. <el-table-column prop="director" />
  888. </el-table-column>
  889. <el-table-column prop="runtime"/>
  890. </el-table>
  891. `,
  892. created() {
  893. this.testData = null;
  894. }
  895. }, true);
  896. setTimeout(_ => {
  897. const trs = vm.$el.querySelectorAll('.el-table__header tr');
  898. expect(trs.length).equal(3);
  899. const firstRowHeader = trs[0].querySelectorAll('th .cell').length;
  900. const secondRowHeader = trs[1].querySelectorAll('th .cell').length;
  901. const thirdRowHeader = trs[2].querySelectorAll('th .cell').length;
  902. expect(firstRowHeader).to.equal(3);
  903. expect(secondRowHeader).to.equal(2);
  904. expect(thirdRowHeader).to.equal(2);
  905. expect(trs[0].querySelector('th:first-child').getAttribute('rowspan')).to.equal('3');
  906. expect(trs[0].querySelector('th:nth-child(2)').getAttribute('colspan')).to.equal('3');
  907. expect(trs[1].querySelector('th:first-child').getAttribute('colspan')).to.equal('2');
  908. expect(trs[1].querySelector('th:nth-child(2)').getAttribute('rowspan')).to.equal('2');
  909. destroyVM(vm);
  910. done();
  911. }, DELAY);
  912. });
  913. it('should work in one column', done => {
  914. const vm = createVue({
  915. template: `
  916. <el-table :data="testData">
  917. <el-table-column label="group">
  918. <el-table-column prop="release"/>
  919. </el-table-column>
  920. </el-table>
  921. `,
  922. created() {
  923. this.testData = null;
  924. }
  925. }, true);
  926. setTimeout(_ => {
  927. const trs = vm.$el.querySelectorAll('.el-table__header tr');
  928. expect(trs.length).equal(2);
  929. const firstRowLength = trs[0].querySelectorAll('th .cell').length;
  930. const secondRowLength = trs[1].querySelectorAll('th .cell').length;
  931. expect(firstRowLength).to.equal(1);
  932. expect(secondRowLength).to.equal(1);
  933. expect(trs[0].querySelector('th:first-child').getAttribute('rowspan')).to.equal('1');
  934. expect(trs[0].querySelector('th:first-child').getAttribute('colspan')).to.equal('1');
  935. destroyVM(vm);
  936. done();
  937. }, DELAY);
  938. });
  939. });
  940. describe('dynamic column attribtes', () => {
  941. const DELAY = 50;
  942. it('label', (done) => {
  943. const vm = createVue({
  944. template: `
  945. <el-table :data="testData">
  946. <el-table-column prop="name" :label="label"/>
  947. <el-table-column prop="release" />
  948. <el-table-column prop="director" />
  949. <el-table-column prop="runtime" />
  950. </el-table>
  951. `,
  952. data() {
  953. return {
  954. label: 'name'
  955. };
  956. },
  957. created() {
  958. this.testData = getTestData();
  959. }
  960. }, true);
  961. setTimeout(() => {
  962. expect(vm.$el.querySelector('.el-table__header th .cell').textContent).to.equal('name');
  963. vm.label = 'NAME';
  964. vm.$nextTick(() => {
  965. expect(vm.$el.querySelector('.el-table__header th .cell').textContent).to.equal('NAME');
  966. });
  967. done();
  968. }, DELAY);
  969. });
  970. it('align', (done) => {
  971. const vm = createVue({
  972. template: `
  973. <el-table :data="testData">
  974. <el-table-column prop="name" :align="align"/>
  975. </el-table>
  976. `,
  977. data() {
  978. return {
  979. align: 'left'
  980. };
  981. },
  982. created() {
  983. this.testData = getTestData();
  984. }
  985. }, true);
  986. setTimeout(() => {
  987. expect(vm.$el.querySelectorAll('.el-table__body td.is-right').length === 0).to.be.true;
  988. vm.align = 'right';
  989. vm.$nextTick(() => {
  990. expect(vm.$el.querySelectorAll('.el-table__body td.is-right').length > 0).to.be.true;
  991. });
  992. done();
  993. }, DELAY);
  994. });
  995. it('width', (done) => {
  996. const vm = createVue({
  997. template: `
  998. <el-table :data="testData" :fit="false">
  999. <el-table-column prop="name" :width="width"/>
  1000. </el-table>
  1001. `,
  1002. data() {
  1003. return {
  1004. width: 100
  1005. };
  1006. },
  1007. created() {
  1008. this.testData = getTestData();
  1009. }
  1010. }, true);
  1011. setTimeout(() => {
  1012. expect(vm.$el.querySelector('.el-table__body col').getAttribute('width')).to.equal('100');
  1013. vm.width = 200;
  1014. vm.$nextTick(() => {
  1015. expect(vm.$el.querySelector('.el-table__body col').getAttribute('width')).to.equal('200');
  1016. });
  1017. done();
  1018. }, DELAY);
  1019. });
  1020. it('min-width', (done) => {
  1021. const vm = createVue({
  1022. template: `
  1023. <el-table :data="testData" :fit="false">
  1024. <el-table-column prop="name" :min-width="width"/>
  1025. </el-table>
  1026. `,
  1027. data() {
  1028. return {
  1029. width: 100
  1030. };
  1031. },
  1032. created() {
  1033. this.testData = getTestData();
  1034. }
  1035. }, true);
  1036. setTimeout(() => {
  1037. expect(vm.$el.querySelector('.el-table__body col').getAttribute('width')).to.equal('100');
  1038. vm.width = 200;
  1039. vm.$nextTick(() => {
  1040. expect(vm.$el.querySelector('.el-table__body col').getAttribute('width')).to.equal('200');
  1041. });
  1042. done();
  1043. }, DELAY);
  1044. });
  1045. it('fixed', (done) => {
  1046. const vm = createVue({
  1047. template: `
  1048. <el-table :data="testData">
  1049. <el-table-column :fixed="fixed" />
  1050. <el-table-column prop="release" />
  1051. <el-table-column prop="director" />
  1052. <el-table-column prop="runtime" />
  1053. </el-table>
  1054. `,
  1055. data() {
  1056. return {
  1057. fixed: false
  1058. };
  1059. },
  1060. created() {
  1061. this.testData = getTestData();
  1062. }
  1063. }, true);
  1064. setTimeout(() => {
  1065. expect(!vm.$el.querySelector('.el-table__fixed')).to.be.true;
  1066. vm.fixed = true;
  1067. vm.$nextTick(() => {
  1068. expect(!!vm.$el.querySelector('.el-table__fixed')).to.be.true;
  1069. });
  1070. done();
  1071. }, DELAY);
  1072. });
  1073. it('prop', (done) => {
  1074. const vm = createVue({
  1075. template: `
  1076. <el-table :data="testData">
  1077. <el-table-column :prop="name" />
  1078. <el-table-column prop="release" />
  1079. <el-table-column prop="director" />
  1080. <el-table-column prop="runtime" />
  1081. </el-table>
  1082. `,
  1083. data() {
  1084. return {
  1085. prop: 'name'
  1086. };
  1087. },
  1088. created() {
  1089. this.testData = getTestData();
  1090. }
  1091. }, true);
  1092. setTimeout(() => {
  1093. let firstColumnContent = vm.$el.querySelector('.el-table__body td .cell').textContent;
  1094. let secondColumnContent = vm.$el.querySelector('.el-table__body td:nth-child(2) .cell').textContent;
  1095. expect(firstColumnContent !== secondColumnContent).to.be.true;
  1096. vm.prop = 'release';
  1097. vm.property = 'release';
  1098. vm.$nextTick(() => {
  1099. firstColumnContent = vm.$el.querySelector('.el-table__body td .cell').textContent;
  1100. secondColumnContent = vm.$el.querySelector('.el-table__body td:nth-child(2) .cell').textContent;
  1101. expect(firstColumnContent === secondColumnContent).to.be.true;
  1102. });
  1103. done();
  1104. }, DELAY);
  1105. });
  1106. });
  1107. describe('methods', () => {
  1108. const createTable = function(prop = '', opts) {
  1109. return createVue({
  1110. template: `
  1111. <el-table ref="table" :data="testData" @${prop}="handleEvent">
  1112. <el-table-column type="selection" />
  1113. <el-table-column prop="name" />
  1114. <el-table-column prop="release" />
  1115. <el-table-column prop="director" />
  1116. <el-table-column prop="runtime"/>
  1117. </el-table>
  1118. `,
  1119. methods: {
  1120. handleEvent(selection) {
  1121. this.fireCount++;
  1122. this.selection = selection;
  1123. }
  1124. },
  1125. created() {
  1126. this.testData = getTestData();
  1127. },
  1128. data() {
  1129. return { selection: null, testData: this.testData, fireCount: 0 };
  1130. }
  1131. }, true);
  1132. };
  1133. it('toggleRowSelection', () => {
  1134. const vm = createTable('selection-change');
  1135. vm.$refs.table.toggleRowSelection(vm.testData[0]);
  1136. expect(vm.selection).to.length(1);
  1137. expect(vm.fireCount).to.equal(1);
  1138. // test use second parameter
  1139. vm.$refs.table.toggleRowSelection(vm.testData[0], true);
  1140. expect(vm.fireCount).to.equal(1);
  1141. vm.$refs.table.toggleRowSelection(vm.testData[0], false);
  1142. expect(vm.fireCount).to.equal(2);
  1143. expect(vm.selection).to.length(0);
  1144. destroyVM(vm);
  1145. });
  1146. it('clearSelection', () => {
  1147. const vm = createTable('selection-change');
  1148. vm.$refs.table.toggleRowSelection(vm.testData[0]);
  1149. expect(vm.selection).to.length(1);
  1150. expect(vm.fireCount).to.equal(1);
  1151. // clear selection
  1152. vm.$refs.table.clearSelection();
  1153. expect(vm.fireCount).to.equal(2);
  1154. expect(vm.selection).to.length(0);
  1155. vm.$refs.table.clearSelection();
  1156. expect(vm.fireCount).to.equal(2);
  1157. destroyVM(vm);
  1158. });
  1159. });
  1160. it('hover', done => {
  1161. const vm = createVue({
  1162. template: `
  1163. <el-table :data="testData">
  1164. <el-table-column prop="name" label="片名" fixed />
  1165. <el-table-column prop="release" label="发行日期" />
  1166. <el-table-column prop="director" label="导演" />
  1167. <el-table-column prop="runtime" label="时长(分)" />
  1168. </el-table>
  1169. `,
  1170. created() {
  1171. this.testData = getTestData();
  1172. }
  1173. }, true);
  1174. setTimeout(_ => {
  1175. const tr = vm.$el.querySelector('.el-table__body-wrapper tbody tr');
  1176. triggerEvent(tr, 'mouseenter', true, false);
  1177. setTimeout(_ => {
  1178. expect(tr.classList.contains('hover-row')).to.true;
  1179. triggerEvent(tr, 'mouseleave', true, false);
  1180. setTimeout(_ => {
  1181. expect(tr.classList.contains('hover-row')).to.false;
  1182. destroyVM(vm);
  1183. done();
  1184. }, DELAY);
  1185. }, DELAY);
  1186. }, DELAY);
  1187. });
  1188. it('highlight-current-row', done => {
  1189. const vm = createVue({
  1190. template: `
  1191. <el-table :data="testData" highlight-current-row>
  1192. <el-table-column prop="name" label="片名" />
  1193. <el-table-column prop="release" label="发行日期" />
  1194. <el-table-column prop="director" label="导演" />
  1195. <el-table-column prop="runtime" label="时长(分)" />
  1196. </el-table>
  1197. `,
  1198. created() {
  1199. this.testData = getTestData();
  1200. }
  1201. }, true);
  1202. setTimeout(_ => {
  1203. const tr = vm.$el.querySelector('.el-table__body-wrapper tbody tr');
  1204. triggerEvent(tr, 'click', true, false);
  1205. setTimeout(_ => {
  1206. expect(tr.classList.contains('current-row')).to.be.true;
  1207. const rows = vm.$el.querySelectorAll('.el-table__body-wrapper tbody tr');
  1208. triggerEvent(rows[2], 'click', true, false);
  1209. setTimeout(_ => {
  1210. expect(tr.classList.contains('current-row')).to.be.false;
  1211. expect(rows[2].classList.contains('current-row')).to.be.true;
  1212. destroyVM(vm);
  1213. done();
  1214. }, DELAY);
  1215. }, DELAY);
  1216. }, DELAY);
  1217. });
  1218. });