widgetmanager.spec.ts 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336
  1. // Copyright (c) Jupyter Development Team.
  2. // Distributed under the terms of the Modified BSD License.
  3. import 'jest';
  4. import { ServiceManager } from '@jupyterlab/services';
  5. import { DocumentWidgetManager } from '../src';
  6. import {
  7. DocumentRegistry,
  8. TextModelFactory,
  9. ABCWidgetFactory,
  10. Context,
  11. DocumentWidget,
  12. IDocumentWidget
  13. } from '@jupyterlab/docregistry';
  14. import { PromiseDelegate, UUID } from '@lumino/coreutils';
  15. import { IMessageHandler, Message, MessageLoop } from '@lumino/messaging';
  16. import { Widget } from '@lumino/widgets';
  17. import { acceptDialog, dismissDialog } from '@jupyterlab/testutils';
  18. import * as Mock from '@jupyterlab/testutils/lib/mock';
  19. class WidgetFactory extends ABCWidgetFactory<IDocumentWidget> {
  20. protected createNewWidget(
  21. context: DocumentRegistry.Context
  22. ): IDocumentWidget {
  23. const content = new Widget();
  24. const widget = new DocumentWidget({ content, context });
  25. widget.addClass('WidgetFactory');
  26. return widget;
  27. }
  28. }
  29. class LoggingManager extends DocumentWidgetManager {
  30. methods: string[] = [];
  31. messageHook(handler: IMessageHandler, msg: Message): boolean {
  32. this.methods.push('messageHook');
  33. return super.messageHook(handler, msg);
  34. }
  35. setCaption(widget: Widget): Promise<void> {
  36. this.methods.push('setCaption');
  37. return super.setCaption(widget);
  38. }
  39. onClose(widget: Widget): Promise<boolean> {
  40. this.methods.push('onClose');
  41. return super.onClose(widget);
  42. }
  43. }
  44. describe('@jupyterlab/docmanager', () => {
  45. let manager: LoggingManager;
  46. let services: ServiceManager.IManager;
  47. const textModelFactory = new TextModelFactory();
  48. let context: Context<DocumentRegistry.IModel>;
  49. const widgetFactory = new WidgetFactory({
  50. name: 'test',
  51. fileTypes: ['text']
  52. });
  53. const readOnlyFactory = new WidgetFactory({
  54. name: 'readonly',
  55. fileTypes: ['text'],
  56. readOnly: true
  57. });
  58. beforeAll(() => {
  59. services = new Mock.ServiceManagerMock();
  60. });
  61. beforeEach(() => {
  62. const registry = new DocumentRegistry({ textModelFactory });
  63. registry.addWidgetFactory(widgetFactory);
  64. manager = new LoggingManager({ registry });
  65. context = new Context({
  66. manager: services,
  67. factory: textModelFactory,
  68. path: UUID.uuid4()
  69. });
  70. });
  71. afterEach(() => {
  72. manager.dispose();
  73. context.dispose();
  74. });
  75. describe('DocumentWidgetManager', () => {
  76. describe('#constructor()', () => {
  77. it('should create a new document widget manager', () => {
  78. expect(manager).toBeInstanceOf(DocumentWidgetManager);
  79. });
  80. });
  81. describe('#isDisposed', () => {
  82. it('should test whether the manager is disposed', () => {
  83. expect(manager.isDisposed).toBe(false);
  84. manager.dispose();
  85. expect(manager.isDisposed).toBe(true);
  86. });
  87. });
  88. describe('#dispose()', () => {
  89. it('should dispose of the resources used by the manager', () => {
  90. expect(manager.isDisposed).toBe(false);
  91. manager.dispose();
  92. expect(manager.isDisposed).toBe(true);
  93. manager.dispose();
  94. expect(manager.isDisposed).toBe(true);
  95. });
  96. });
  97. describe('#createWidget()', () => {
  98. it('should create a widget', () => {
  99. const widget = manager.createWidget(widgetFactory, context);
  100. expect(widget).toBeInstanceOf(Widget);
  101. });
  102. it('should emit the widgetCreated signal', () => {
  103. let called = false;
  104. widgetFactory.widgetCreated.connect(() => {
  105. called = true;
  106. });
  107. manager.createWidget(widgetFactory, context);
  108. expect(called).toBe(true);
  109. });
  110. });
  111. describe('#adoptWidget()', () => {
  112. it('should install a message hook', () => {
  113. const content = new Widget();
  114. const widget = new DocumentWidget({ content, context });
  115. manager.adoptWidget(context, widget);
  116. MessageLoop.sendMessage(widget, new Message('foo'));
  117. expect(manager.methods).toEqual(
  118. expect.arrayContaining(['messageHook'])
  119. );
  120. });
  121. it('should add the document class', () => {
  122. const content = new Widget();
  123. const widget = new DocumentWidget({ content, context });
  124. manager.adoptWidget(context, widget);
  125. expect(widget.hasClass('jp-Document')).toBe(true);
  126. });
  127. it('should be retrievable', () => {
  128. const content = new Widget();
  129. const widget = new DocumentWidget({ content, context });
  130. manager.adoptWidget(context, widget);
  131. expect(manager.contextForWidget(widget)).toBe(context);
  132. });
  133. });
  134. describe('#findWidget()', () => {
  135. it('should find a registered widget', () => {
  136. const widget = manager.createWidget(widgetFactory, context);
  137. expect(manager.findWidget(context, 'test')).toBe(widget);
  138. });
  139. it('should return undefined if not found', () => {
  140. expect(manager.findWidget(context, 'test')).toBeUndefined();
  141. });
  142. });
  143. describe('#contextForWidget()', () => {
  144. it('should return the context for a widget', () => {
  145. const widget = manager.createWidget(widgetFactory, context);
  146. expect(manager.contextForWidget(widget)).toBe(context);
  147. });
  148. it('should return undefined if not tracked', () => {
  149. expect(manager.contextForWidget(new Widget())).toBeUndefined();
  150. });
  151. });
  152. describe('#cloneWidget()', () => {
  153. it('should create a new widget with the same context using the same factory', () => {
  154. const widget = manager.createWidget(widgetFactory, context);
  155. const clone = manager.cloneWidget(widget)!;
  156. expect(clone.hasClass('WidgetFactory')).toBe(true);
  157. expect(clone.hasClass('jp-Document')).toBe(true);
  158. expect(manager.contextForWidget(clone)).toBe(context);
  159. });
  160. it('should return undefined if the source widget is not managed', () => {
  161. expect(manager.cloneWidget(new Widget())).toBeUndefined();
  162. });
  163. });
  164. describe('#closeWidgets()', () => {
  165. it('should close all of the widgets associated with a context', async () => {
  166. const widget = manager.createWidget(widgetFactory, context);
  167. const clone = manager.cloneWidget(widget)!;
  168. await manager.closeWidgets(context);
  169. expect(widget.isDisposed).toBe(true);
  170. expect(clone.isDisposed).toBe(true);
  171. });
  172. });
  173. describe('#messageHook()', () => {
  174. it('should be called for a message to a tracked widget', () => {
  175. const content = new Widget();
  176. const widget = new DocumentWidget({ content, context });
  177. manager.adoptWidget(context, widget);
  178. MessageLoop.sendMessage(widget, new Message('foo'));
  179. expect(manager.methods).toEqual(
  180. expect.arrayContaining(['messageHook'])
  181. );
  182. });
  183. it('should return false for close-request messages', () => {
  184. const widget = manager.createWidget(widgetFactory, context);
  185. const msg = new Message('close-request');
  186. expect(manager.messageHook(widget, msg)).toBe(false);
  187. });
  188. it('should return true for other messages', () => {
  189. const widget = manager.createWidget(widgetFactory, context);
  190. const msg = new Message('foo');
  191. expect(manager.messageHook(widget, msg)).toBe(true);
  192. });
  193. });
  194. describe('#setCaption()', () => {
  195. it('should set the title of the widget', async () => {
  196. await context.initialize(true);
  197. const widget = manager.createWidget(widgetFactory, context);
  198. const delegate = new PromiseDelegate();
  199. widget.title.changed.connect(async () => {
  200. expect(manager.methods).toEqual(
  201. expect.arrayContaining(['setCaption'])
  202. );
  203. expect(widget.title.caption).toContain('Last Checkpoint');
  204. await dismissDialog();
  205. delegate.resolve(undefined);
  206. });
  207. await delegate.promise;
  208. });
  209. });
  210. describe('#onClose()', () => {
  211. it('should be called when a widget is closed', async () => {
  212. const widget = manager.createWidget(widgetFactory, context);
  213. const delegate = new PromiseDelegate();
  214. widget.disposed.connect(async () => {
  215. expect(manager.methods).toEqual(expect.arrayContaining(['onClose']));
  216. await dismissDialog();
  217. delegate.resolve(undefined);
  218. });
  219. widget.close();
  220. await delegate.promise;
  221. });
  222. it('should prompt the user before closing', async () => {
  223. // Populate the model with content.
  224. context.model.fromString('foo');
  225. const widget = manager.createWidget(widgetFactory, context);
  226. const closed = manager.onClose(widget);
  227. await Promise.all([acceptDialog(), closed]);
  228. expect(widget.isDisposed).toBe(true);
  229. });
  230. it('should not prompt if the factory is readonly', async () => {
  231. const readonly = manager.createWidget(readOnlyFactory, context);
  232. await manager.onClose(readonly);
  233. expect(readonly.isDisposed).toBe(true);
  234. });
  235. it('should not prompt if the other widget is writable', async () => {
  236. // Populate the model with content.
  237. context.model.fromString('foo');
  238. const one = manager.createWidget(widgetFactory, context);
  239. const two = manager.createWidget(widgetFactory, context);
  240. await manager.onClose(one);
  241. expect(one.isDisposed).toBe(true);
  242. expect(two.isDisposed).toBe(false);
  243. two.dispose();
  244. });
  245. it('should prompt if the only other widget has a readonly factory', async () => {
  246. // Populate the model with content.
  247. context.model.fromString('foo');
  248. const writable = manager.createWidget(widgetFactory, context);
  249. const readonly = manager.createWidget(readOnlyFactory, context);
  250. const closed = manager.onClose(writable);
  251. await acceptDialog();
  252. await closed;
  253. expect(writable.isDisposed).toBe(true);
  254. expect(readonly.isDisposed).toBe(false);
  255. readonly.dispose();
  256. });
  257. it('should close the widget', async () => {
  258. context.model.fromString('foo');
  259. const widget = manager.createWidget(widgetFactory, context);
  260. const promise = manager.onClose(widget);
  261. await dismissDialog();
  262. await promise;
  263. expect(widget.isDisposed).toBe(false);
  264. });
  265. });
  266. });
  267. });