widget.spec.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  1. // Copyright (c) Jupyter Development Team.
  2. // Distributed under the terms of the Modified BSD License.
  3. import expect = require('expect.js');
  4. import {
  5. Message
  6. } from 'phosphor-messaging';
  7. import {
  8. ChildMessage, Widget
  9. } from 'phosphor-widget';
  10. import {
  11. nbformat
  12. } from '../../../../lib/notebook/notebook/nbformat';
  13. import {
  14. OutputAreaModel, OutputAreaWidget
  15. } from '../../../../lib/notebook/output-area';
  16. import {
  17. MimeMap, RenderMime
  18. } from '../../../../lib/rendermime';
  19. import {
  20. defaultRenderMime
  21. } from '../../rendermime/rendermime.spec';
  22. import {
  23. DEFAULT_OUTPUTS
  24. } from './model.spec';
  25. /**
  26. * The default rendermime instance to use for testing.
  27. */
  28. const rendermime = defaultRenderMime();
  29. class LogOutputAreaWidget extends OutputAreaWidget {
  30. methods: string[] = [];
  31. protected onUpdateRequest(msg: Message): void {
  32. super.onUpdateRequest(msg);
  33. this.methods.push('onUpdateRequest');
  34. }
  35. protected onChildAdded(msg: ChildMessage): void {
  36. super.onChildAdded(msg);
  37. this.methods.push('onChildAdded');
  38. }
  39. protected onChildRemoved(msg: ChildMessage): void {
  40. super.onChildRemoved(msg);
  41. this.methods.push('onChildRemoved');
  42. }
  43. protected onModelChanged(oldValue: OutputAreaModel, newValue: OutputAreaModel): void {
  44. super.onModelChanged(oldValue, newValue);
  45. this.methods.push('onModelChanged');
  46. }
  47. }
  48. class LogRenderer extends OutputAreaWidget.Renderer {
  49. methods: string[] = [];
  50. createOutput(): Widget {
  51. this.methods.push('createOutput');
  52. return super.createOutput();
  53. }
  54. updateOutput(output: nbformat.IOutput, rendermime: RenderMime<Widget>, widget: Widget, trusted?: boolean): void {
  55. super.updateOutput(output, rendermime, widget, trusted);
  56. this.methods.push('updateOutput');
  57. }
  58. }
  59. function createWidget(): LogOutputAreaWidget {
  60. let widget = new LogOutputAreaWidget({ rendermime });
  61. let model = new OutputAreaModel();
  62. for (let output of DEFAULT_OUTPUTS) {
  63. model.add(output);
  64. }
  65. widget.model = model;
  66. return widget;
  67. }
  68. describe('notebook/output-area/widget', () => {
  69. describe('OutputAreaWidget', () => {
  70. describe('#constructor()', () => {
  71. it('should take an options object', () => {
  72. let widget = new OutputAreaWidget({ rendermime });
  73. expect(widget).to.be.an(OutputAreaWidget);
  74. });
  75. it('should take an optional renderer', () => {
  76. let renderer = Object.create(OutputAreaWidget.defaultRenderer);
  77. let widget = new OutputAreaWidget({ rendermime, renderer });
  78. expect(widget.renderer).to.be(renderer);
  79. });
  80. it('should add the `jp-OutputArea` class', () => {
  81. let widget = new OutputAreaWidget({ rendermime });
  82. expect(widget.hasClass('jp-OutputArea')).to.be(true);
  83. });
  84. });
  85. describe('#modelChanged', () => {
  86. it('should be emitted when the model of the widget changes', () => {
  87. let widget = new OutputAreaWidget({ rendermime });
  88. let called = false;
  89. widget.modelChanged.connect((sender, args) => {
  90. expect(sender).to.be(widget);
  91. expect(args).to.be(void 0);
  92. called = true;
  93. });
  94. widget.model = new OutputAreaModel();
  95. expect(called).to.be(true);
  96. });
  97. });
  98. describe('#model', () => {
  99. it('should default to `null`', () => {
  100. let widget = new OutputAreaWidget({ rendermime });
  101. expect(widget.model).to.be(null);
  102. });
  103. it('should set the model', () => {
  104. let widget = new OutputAreaWidget({ rendermime });
  105. let model = new OutputAreaModel();
  106. widget.model = model;
  107. expect(widget.model).to.be(model);
  108. });
  109. it('should emit `modelChanged` when the model changes', () => {
  110. let widget = new OutputAreaWidget({ rendermime });
  111. let called = false;
  112. widget.modelChanged.connect(() => { called = true; });
  113. widget.model = new OutputAreaModel();
  114. expect(called).to.be(true);
  115. });
  116. it('should not emit `modelChanged` when the model does not change', () => {
  117. let widget = new OutputAreaWidget({ rendermime });
  118. let called = false;
  119. let model = new OutputAreaModel();
  120. widget.model = model;
  121. widget.modelChanged.connect(() => { called = true; });
  122. widget.model = model;
  123. expect(called).to.be(false);
  124. });
  125. it('should create widgets for the model items', () => {
  126. let widget = createWidget();
  127. expect(widget.childCount()).to.be(5);
  128. });
  129. context('model `changed` signal', () => {
  130. it('should dispose of the child widget when an output is removed', () => {
  131. let widget = createWidget();
  132. let child = widget.childAt(0);
  133. widget.model.clear();
  134. expect(child.isDisposed).to.be(true);
  135. });
  136. it('should add the `jp-OutputArea-output` class to new widgets', () => {
  137. let widget = new LogOutputAreaWidget({ rendermime });
  138. widget.model = new OutputAreaModel();
  139. widget.model.add(DEFAULT_OUTPUTS[0]);
  140. let child = widget.childAt(0);
  141. expect(child.hasClass('jp-OutputArea-output')).to.be(true);
  142. });
  143. });
  144. });
  145. describe('#rendermime', () => {
  146. it('should be the rendermime instance used by the widget', () => {
  147. let widget = new OutputAreaWidget({ rendermime });
  148. expect(widget.rendermime).to.be(rendermime);
  149. });
  150. it('should be read-only', () => {
  151. let widget = new OutputAreaWidget({ rendermime });
  152. expect(() => { widget.rendermime = null; }).to.throwError();
  153. });
  154. });
  155. describe('#renderer', () => {
  156. it('should be the renderer used by the widget', () => {
  157. let renderer = new OutputAreaWidget.Renderer();
  158. let widget = new OutputAreaWidget({ rendermime, renderer });
  159. expect(widget.renderer).to.be(renderer);
  160. });
  161. it('should be read-only', () => {
  162. let widget = new OutputAreaWidget({ rendermime });
  163. expect(() => { widget.renderer = null; }).to.throwError();
  164. });
  165. it('should be called to create and update when a widget is added', () => {
  166. let renderer = new LogRenderer();
  167. let widget = new LogOutputAreaWidget({ rendermime, renderer });
  168. let model = new OutputAreaModel();
  169. widget.model = model;
  170. model.add(DEFAULT_OUTPUTS[0]);
  171. expect(renderer.methods).to.contain('createOutput');
  172. expect(renderer.methods).to.contain('updateOutput');
  173. });
  174. });
  175. describe('#trusted', () => {
  176. it('should get the trusted state of the widget', () => {
  177. let widget = new OutputAreaWidget({ rendermime });
  178. expect(widget.trusted).to.be(false);
  179. });
  180. it('should set the trusted state of the widget', () => {
  181. let widget = new OutputAreaWidget({ rendermime });
  182. widget.trusted = true;
  183. expect(widget.trusted).to.be(true);
  184. });
  185. it('should re-render the widgets', () => {
  186. let renderer = new LogRenderer();
  187. let widget = new OutputAreaWidget({ rendermime, renderer });
  188. widget.model = new OutputAreaModel();
  189. widget.model.add(DEFAULT_OUTPUTS[0]);
  190. renderer.methods = [];
  191. widget.trusted = true;
  192. expect(renderer.methods).to.contain('updateOutput');
  193. });
  194. });
  195. describe('#collapsed', () => {
  196. it('should get the collapsed state of the widget', () => {
  197. let widget = createWidget();
  198. expect(widget.collapsed).to.be(false);
  199. });
  200. it('should set the collapsed state of the widget', () => {
  201. let widget = createWidget();
  202. widget.collapsed = true;
  203. expect(widget.collapsed).to.be(true);
  204. });
  205. it('should post an update request', (done) => {
  206. let widget = new LogOutputAreaWidget({ rendermime });
  207. widget.collapsed = true;
  208. requestAnimationFrame(() => {
  209. expect(widget.methods).to.contain('onUpdateRequest');
  210. done();
  211. });
  212. });
  213. });
  214. describe('#fixedHeight', () => {
  215. it('should get the fixed height state of the widget', () => {
  216. let widget = createWidget();
  217. expect(widget.fixedHeight).to.be(false);
  218. });
  219. it('should set the fixed height state of the widget', () => {
  220. let widget = createWidget();
  221. widget.fixedHeight = true;
  222. expect(widget.fixedHeight).to.be(true);
  223. });
  224. it('should post an update request', (done) => {
  225. let widget = new LogOutputAreaWidget({ rendermime });
  226. widget.fixedHeight = true;
  227. requestAnimationFrame(() => {
  228. expect(widget.methods).to.contain('onUpdateRequest');
  229. done();
  230. });
  231. });
  232. });
  233. describe('#dispose()', () => {
  234. it('should dispose of the resources held by the widget', () => {
  235. let widget = createWidget();
  236. widget.dispose();
  237. expect(widget.model).to.be(null);
  238. expect(widget.rendermime).to.be(null);
  239. expect(widget.renderer).to.be(null);
  240. });
  241. it('should be safe to call more than once', () => {
  242. let widget = createWidget();
  243. widget.dispose();
  244. widget.dispose();
  245. expect(widget.isDisposed).to.be(true);
  246. });
  247. });
  248. describe('#childAt()', () => {
  249. it('should get the child widget at the specified index', () => {
  250. let widget = createWidget();
  251. expect(widget.childAt(0)).to.be.a(Widget);
  252. });
  253. });
  254. describe('#childCount()', () => {
  255. it('should get the number of child widgets', () => {
  256. let widget = createWidget();
  257. expect(widget.childCount()).to.be(5);
  258. widget.model.clear();
  259. expect(widget.childCount()).to.be(0);
  260. });
  261. });
  262. describe('#onUpdateRequest()', () => {
  263. it('should set the appropriate classes on the widget', (done) => {
  264. let widget = createWidget();
  265. widget.collapsed = true;
  266. widget.fixedHeight = true;
  267. requestAnimationFrame(() => {
  268. expect(widget.methods).to.contain('onUpdateRequest');
  269. expect(widget.hasClass('jp-mod-fixedHeight')).to.be(true);
  270. expect(widget.hasClass('jp-mod-collapsed')).to.be(true);
  271. done();
  272. });
  273. });
  274. });
  275. describe('#onModelChanged()', () => {
  276. it('should be called when the model changes', () => {
  277. let widget = new LogOutputAreaWidget({ rendermime });
  278. widget.model = new OutputAreaModel();
  279. expect(widget.methods).to.contain('onModelChanged');
  280. });
  281. it('should not be called when the model does not change', () => {
  282. let widget = new LogOutputAreaWidget({ rendermime });
  283. widget.model = new OutputAreaModel();
  284. widget.methods = [];
  285. widget.model = widget.model;
  286. expect(widget.methods).to.not.contain('onModelChanged');
  287. });
  288. });
  289. describe('.Renderer', () => {
  290. describe('#createOutput()', () => {
  291. it('should create a widget', () => {
  292. let renderer = new OutputAreaWidget.Renderer();
  293. let widget = renderer.createOutput();
  294. expect(widget).to.be.a(Widget);
  295. });
  296. });
  297. describe('#updateOutput()', () => {
  298. it('should handle all bundle types when trusted', () => {
  299. let renderer = new OutputAreaWidget.Renderer();
  300. let widget = renderer.createOutput();
  301. for (let i = 0; i < DEFAULT_OUTPUTS.length; i++) {
  302. let output = DEFAULT_OUTPUTS[i];
  303. renderer.updateOutput(output, rendermime, widget, true);
  304. }
  305. });
  306. it('should handle all bundle types when not trusted', () => {
  307. let renderer = new OutputAreaWidget.Renderer();
  308. let widget = renderer.createOutput();
  309. for (let i = 0; i < DEFAULT_OUTPUTS.length; i++) {
  310. let output = DEFAULT_OUTPUTS[i];
  311. renderer.updateOutput(output, rendermime, widget, false);
  312. }
  313. });
  314. });
  315. describe('#getBundle()', () => {
  316. it('should handle all bundle types', () => {
  317. let renderer = new OutputAreaWidget.Renderer();
  318. for (let i = 0; i < DEFAULT_OUTPUTS.length; i++) {
  319. let output = DEFAULT_OUTPUTS[i];
  320. let bundle = renderer.getBundle(output);
  321. expect(Object.keys(bundle).length).to.not.be(0);
  322. }
  323. });
  324. });
  325. describe('#convertBundle()', () => {
  326. it('should handle bundles with strings', () => {
  327. let bundle: nbformat.MimeBundle = {
  328. 'text/plain': 'foo'
  329. };
  330. let renderer = new OutputAreaWidget.Renderer();
  331. let map = renderer.convertBundle(bundle);
  332. expect(map).to.eql(bundle);
  333. });
  334. it('should handle bundles with string arrays', () => {
  335. let bundle: nbformat.MimeBundle = {
  336. 'text/plain': ['foo', 'bar']
  337. };
  338. let renderer = new OutputAreaWidget.Renderer();
  339. let map = renderer.convertBundle(bundle);
  340. expect(map).to.eql({ 'text/plain': 'foo\nbar' });
  341. });
  342. });
  343. describe('#sanitize()', () => {
  344. it('should sanitize html input', () => {
  345. let map: MimeMap<string> = {
  346. 'text/html': '<div>hello, 1 < 2</div>'
  347. };
  348. let renderer = new OutputAreaWidget.Renderer();
  349. renderer.sanitize(map);
  350. expect(map['text/html']).to.be('<div>hello, 1 &lt; 2</div>');
  351. });
  352. it('should allow text/plain', () => {
  353. let map: MimeMap<string> = {
  354. 'text/plain': '<div>hello, 1 < 2</div>'
  355. };
  356. let renderer = new OutputAreaWidget.Renderer();
  357. renderer.sanitize(map);
  358. expect(map['text/plain']).to.be('<div>hello, 1 < 2</div>');
  359. });
  360. it('should disallow unknown mimetype', () => {
  361. let map: MimeMap<string> = {
  362. 'foo/bar': '<div>hello, 1 < 2</div>'
  363. };
  364. let renderer = new OutputAreaWidget.Renderer();
  365. renderer.sanitize(map);
  366. expect(map['foo/bar']).to.be(void 0);
  367. });
  368. });
  369. });
  370. describe('.defaultRenderer', () => {
  371. it('should be a `Renderer` instance', () => {
  372. expect(OutputAreaWidget.defaultRenderer).to.be.an(OutputAreaWidget.Renderer);
  373. });
  374. });
  375. });
  376. });