ListView.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. describe('ListView rendering', function() {
  2. pushOptions({
  3. defaultView: 'listWeek',
  4. now: '2016-08-20'
  5. })
  6. describe('with all-day events', function() {
  7. describe('when single-day', function() {
  8. pushOptions({
  9. events: [
  10. {
  11. title: 'event 1',
  12. start: '2016-08-15'
  13. },
  14. {
  15. title: 'event 2',
  16. start: '2016-08-17'
  17. }
  18. ]
  19. })
  20. it('renders only days with events', function() {
  21. initCalendar()
  22. var days = getDayInfo()
  23. var events = getEventInfo()
  24. expect(days.length).toBe(2)
  25. expect(days[0].date.format()).toEqual('2016-08-15')
  26. expect(days[1].date.format()).toEqual('2016-08-17')
  27. expect(events.length).toBe(2)
  28. expect(events[0].title).toBe('event 1')
  29. expect(events[0].timeText).toBe('all-day')
  30. expect(events[1].title).toBe('event 2')
  31. expect(events[1].timeText).toBe('all-day')
  32. })
  33. it('filters events through eventRender', function() {
  34. var options = {}
  35. options.eventRender = function(event, el) {
  36. el.find('.fc-event-dot').replaceWith('<span class="custom-icon" />')
  37. }
  38. initCalendar(options)
  39. expect($('.custom-icon').length).toBe(2)
  40. })
  41. })
  42. describe('when multi-day', function() {
  43. pushOptions({
  44. events: [
  45. {
  46. title: 'event 1',
  47. start: '2016-08-15',
  48. end: '2016-08-18' // 3 days
  49. }
  50. ]
  51. })
  52. it('renders all-day for every day', function() {
  53. initCalendar()
  54. var events = getEventInfo()
  55. expect(events.length).toBe(3)
  56. expect(events[0].title).toBe('event 1')
  57. expect(events[0].timeText).toBe('all-day')
  58. expect(events[1].title).toBe('event 1')
  59. expect(events[1].timeText).toBe('all-day')
  60. expect(events[2].title).toBe('event 1')
  61. expect(events[2].timeText).toBe('all-day')
  62. })
  63. })
  64. })
  65. describe('with timed events', function() {
  66. describe('when single-day', function() {
  67. pushOptions({
  68. events: [
  69. {
  70. title: 'event 1',
  71. start: '2016-08-15T07:00'
  72. },
  73. {
  74. title: 'event 2',
  75. start: '2016-08-17T09:00',
  76. end: '2016-08-17T11:00'
  77. }
  78. ]
  79. })
  80. it('renders times', function() {
  81. initCalendar()
  82. var events = getEventInfo()
  83. expect(events.length).toBe(2)
  84. expect(events[0].title).toBe('event 1')
  85. expect(events[0].timeText).toBe('7:00am')
  86. expect(events[1].title).toBe('event 2')
  87. expect(events[1].timeText).toBe('9:00am - 11:00am')
  88. })
  89. it('doesn\'t render times when displayEventTime is false', function() {
  90. var options = {}
  91. options.displayEventTime = false
  92. initCalendar(options)
  93. var events = getEventInfo()
  94. expect(events.length).toBe(2)
  95. expect(events[0].title).toBe('event 1')
  96. expect(events[0].timeText).toBe('')
  97. expect(events[1].title).toBe('event 2')
  98. expect(events[1].timeText).toBe('')
  99. })
  100. it('doesn\'t render end times when displayEventEnd is false', function() {
  101. var options = {}
  102. options.displayEventEnd = false
  103. initCalendar(options)
  104. var events = getEventInfo()
  105. expect(events.length).toBe(2)
  106. expect(events[0].title).toBe('event 1')
  107. expect(events[0].timeText).toBe('7:00am')
  108. expect(events[1].title).toBe('event 2')
  109. expect(events[1].timeText).toBe('9:00am')
  110. })
  111. // regression test for when localized event dates get unlocalized and leak into view rendering
  112. it('renders dates and times in locale', function() {
  113. var options = {}
  114. options.locale = 'fr'
  115. initCalendar(options)
  116. var days = getDayInfo()
  117. var events = getEventInfo()
  118. expect(days.length).toBe(2)
  119. expect(days[0].date.format()).toEqual('2016-08-15')
  120. expect(days[0].mainText).toEqual('lundi')
  121. expect(days[0].altText).toEqual('15 août 2016')
  122. expect(days[1].date.format()).toEqual('2016-08-17')
  123. expect(days[1].mainText).toEqual('mercredi')
  124. expect(days[1].altText).toEqual('17 août 2016')
  125. expect(events.length).toBe(2)
  126. expect(events[0].title).toBe('event 1')
  127. expect(events[0].timeText).toBe('07:00')
  128. expect(events[1].title).toBe('event 2')
  129. expect(events[1].timeText).toBe('09:00 - 11:00')
  130. })
  131. })
  132. describe('when multi-day', function() {
  133. pushOptions({
  134. nextDayThreshold: '00:00'
  135. })
  136. it('renders partial and full days', function() {
  137. var options = {}
  138. options.events = [
  139. {
  140. title: 'event 1',
  141. start: '2016-08-15T07:00',
  142. end: '2016-08-17T11:00'
  143. }
  144. ]
  145. initCalendar(options)
  146. var events = getEventInfo()
  147. expect(events.length).toBe(3)
  148. expect(events[0].title).toBe('event 1')
  149. expect(events[0].timeText).toBe('7:00am - 12:00am')
  150. expect(events[1].title).toBe('event 1')
  151. expect(events[1].timeText).toBe('all-day')
  152. expect(events[2].title).toBe('event 1')
  153. expect(events[2].timeText).toBe('12:00am - 11:00am')
  154. })
  155. it('truncates an out-of-range start', function() {
  156. var options = {}
  157. options.events = [
  158. {
  159. title: 'event 1',
  160. start: '2016-08-13T07:00',
  161. end: '2016-08-16T11:00'
  162. }
  163. ]
  164. initCalendar(options)
  165. var events = getEventInfo()
  166. expect(events.length).toBe(3)
  167. expect(events[0].title).toBe('event 1')
  168. expect(events[0].timeText).toBe('all-day')
  169. expect(events[1].title).toBe('event 1')
  170. expect(events[1].timeText).toBe('all-day')
  171. expect(events[2].title).toBe('event 1')
  172. expect(events[2].timeText).toBe('12:00am - 11:00am')
  173. })
  174. it('truncates an out-of-range start', function() {
  175. var options = {}
  176. options.events = [
  177. {
  178. title: 'event 1',
  179. start: '2016-08-18T07:00',
  180. end: '2016-08-21T11:00'
  181. }
  182. ]
  183. initCalendar(options)
  184. var events = getEventInfo()
  185. expect(events.length).toBe(3)
  186. expect(events[0].title).toBe('event 1')
  187. expect(events[0].timeText).toBe('7:00am - 12:00am')
  188. expect(events[1].title).toBe('event 1')
  189. expect(events[1].timeText).toBe('all-day')
  190. expect(events[2].title).toBe('event 1')
  191. expect(events[2].timeText).toBe('all-day')
  192. })
  193. })
  194. it('renders same days when equal to nextDayThreshold', function() {
  195. var options = {}
  196. options.nextDayThreshold = '09:00'
  197. options.events = [
  198. {
  199. title: 'event 1',
  200. start: '2016-08-15T07:00',
  201. end: '2016-08-17T09:00'
  202. }
  203. ]
  204. initCalendar(options)
  205. var events = getEventInfo()
  206. expect(events.length).toBe(3)
  207. expect(events[0].title).toBe('event 1')
  208. expect(events[0].timeText).toBe('7:00am - 12:00am')
  209. expect(events[1].title).toBe('event 1')
  210. expect(events[1].timeText).toBe('all-day')
  211. expect(events[2].title).toBe('event 1')
  212. expect(events[2].timeText).toBe('12:00am - 9:00am')
  213. })
  214. it('renders fewer days when before nextDayThreshold', function() {
  215. var options = {}
  216. options.nextDayThreshold = '09:00'
  217. options.events = [
  218. {
  219. title: 'event 1',
  220. start: '2016-08-15T07:00',
  221. end: '2016-08-17T08:00'
  222. }
  223. ]
  224. initCalendar(options)
  225. var events = getEventInfo()
  226. expect(events.length).toBe(2)
  227. expect(events[0].title).toBe('event 1')
  228. expect(events[0].timeText).toBe('7:00am - 12:00am')
  229. expect(events[1].title).toBe('event 1')
  230. expect(events[1].timeText).toBe('12:00am - 8:00am')
  231. })
  232. })
  233. describe('when an event has no title', function() {
  234. it('renders no text for its title', function() {
  235. var options = {}
  236. options.events = [
  237. {
  238. start: '2016-08-15'
  239. }
  240. ]
  241. initCalendar(options)
  242. var events = getEventInfo()
  243. expect(events.length).toBe(1)
  244. expect(events[0].title).toBe('')
  245. expect(events[0].timeText).toBe('all-day')
  246. })
  247. })
  248. describe('when no events', function() {
  249. it('renders an empty message', function() {
  250. initCalendar()
  251. expect(getIsEmptyMessage()).toBe(true)
  252. })
  253. })
  254. describe('with lots of events', function() {
  255. pushOptions({
  256. now: '2016-08-29',
  257. events: [
  258. {
  259. title: 'All Day Event',
  260. start: '2016-08-29'
  261. },
  262. {
  263. title: 'Long Event',
  264. start: '2016-08-28',
  265. end: '2016-09-04'
  266. },
  267. {
  268. title: 'Meeting',
  269. start: '2016-08-29T10:30:00'
  270. },
  271. {
  272. title: 'Lunch',
  273. start: '2016-08-30T12:00:00'
  274. },
  275. {
  276. title: 'Meeting',
  277. start: '2016-08-30T14:30:00'
  278. },
  279. {
  280. title: 'Happy Hour',
  281. start: '2014-11-12T17:30:00'
  282. },
  283. {
  284. title: 'Dinner',
  285. start: '2014-11-12T20:00:00'
  286. },
  287. {
  288. title: 'Birthday Party',
  289. start: '2016-08-29T07:00:00'
  290. },
  291. {
  292. title: 'Click for Google',
  293. url: 'http://google.com/',
  294. start: '2016-08-31'
  295. }
  296. ]
  297. })
  298. it('sorts events correctly', function() {
  299. initCalendar()
  300. var days = getDayInfo()
  301. var events = getEventInfo()
  302. expect(days.length).toBe(7)
  303. expect(days[0].date.format()).toEqual('2016-08-28')
  304. expect(days[1].date.format()).toEqual('2016-08-29')
  305. expect(days[2].date.format()).toEqual('2016-08-30')
  306. expect(days[3].date.format()).toEqual('2016-08-31')
  307. expect(days[4].date.format()).toEqual('2016-09-01')
  308. expect(days[5].date.format()).toEqual('2016-09-02')
  309. expect(days[6].date.format()).toEqual('2016-09-03')
  310. expect(events.length).toBe(13)
  311. expect(events[0].title).toBe('Long Event')
  312. expect(events[0].timeText).toBe('all-day')
  313. expect(events[1].title).toBe('Long Event')
  314. expect(events[1].timeText).toBe('all-day')
  315. expect(events[2].title).toBe('All Day Event')
  316. expect(events[2].timeText).toBe('all-day')
  317. expect(events[3].title).toBe('Birthday Party')
  318. expect(events[3].timeText).toBe('7:00am')
  319. expect(events[4].title).toBe('Meeting')
  320. expect(events[4].timeText).toBe('10:30am')
  321. expect(events[5].title).toBe('Long Event')
  322. expect(events[5].timeText).toBe('all-day')
  323. expect(events[6].title).toBe('Lunch')
  324. expect(events[6].timeText).toBe('12:00pm')
  325. expect(events[7].title).toBe('Meeting')
  326. expect(events[7].timeText).toBe('2:30pm')
  327. expect(events[8].title).toBe('Long Event')
  328. expect(events[8].timeText).toBe('all-day')
  329. expect(events[9].title).toBe('Click for Google')
  330. expect(events[9].timeText).toBe('all-day')
  331. expect(events[10].title).toBe('Long Event')
  332. expect(events[10].timeText).toBe('all-day')
  333. expect(events[11].title).toBe('Long Event')
  334. expect(events[11].timeText).toBe('all-day')
  335. expect(events[12].title).toBe('Long Event')
  336. expect(events[12].timeText).toBe('all-day')
  337. })
  338. it('makes scrollbars', function() {
  339. let $el = $('<div style="width:300px" />').appendTo('body')
  340. initCalendar({
  341. header: false
  342. }, $el)
  343. let $scrollEl = $('.fc-view .fc-scroller')
  344. expect(
  345. $scrollEl[0].scrollHeight
  346. ).toBeGreaterThan(
  347. $scrollEl[0].clientHeight + 100
  348. )
  349. $el.remove()
  350. })
  351. it('doesn\'t have scrollbars when height is \'auto\'', function() {
  352. let $el = $('<div style="width:300px" />').appendTo('body')
  353. initCalendar({
  354. header: false,
  355. height: 'auto'
  356. }, $el)
  357. let $scrollEl = $('.fc-view .fc-scroller')
  358. expect(
  359. Math.abs($scrollEl[0].scrollHeight - $scrollEl[0].clientHeight)
  360. ).toBeLessThan(2)
  361. $el.remove()
  362. })
  363. })
  364. it('updates rendered events despite fetch range being lazy', function() {
  365. var options = {}
  366. options.now = '2016-09-12'
  367. options.defaultView = 'month'
  368. options.events = [
  369. { title: 'event1', start: '2016-09-12' }
  370. ]
  371. initCalendar(options)
  372. currentCalendar.changeView('listWeek')
  373. expect($('.fc-list-item').length).toBe(1)
  374. currentCalendar.prev()
  375. expect($('.fc-list-item').length).toBe(0)
  376. })
  377. function getDayInfo() {
  378. return $('.fc-list-heading').map(function(i, el) {
  379. el = $(el)
  380. return {
  381. mainText: el.find('.fc-list-heading-main').text() || '',
  382. altText: el.find('.fc-list-heading-alt').text() || '',
  383. date: $.fullCalendar.moment(el.data('date'))
  384. }
  385. }).get()
  386. }
  387. function getEventInfo() { // gets all *segments*
  388. return $('.fc-list-item').map(function(i, el) {
  389. el = $(el)
  390. return {
  391. title: el.find('.fc-list-item-title').text() || '', // text!
  392. timeText: el.find('.fc-list-item-time').text() || '' // text!
  393. }
  394. }).get()
  395. }
  396. function getIsEmptyMessage() {
  397. return Boolean($('.fc-list-empty').length)
  398. }
  399. })