packages/guides/src/GuidesManager.tsx

  1. import { ref, Properties } from "framework-utils";
  2. import * as React from "react";
  3. import { PROPERTIES, METHODS, EVENTS } from "./consts";
  4. import { GuidesInterface, GuidesEvents, GuidesOptions } from "@scena/react-guides/declaration/types";
  5. import InnerGuides from "./InnerGuides";
  6. import EventEmitter from "@scena/event-emitter";
  7. import { camelize } from "@daybrush/utils";
  8. import { ContainerProvider, renderSelf } from "croact";
  9. @Properties(METHODS as any, (prototype, property) => {
  10. if (prototype[property]) {
  11. return;
  12. }
  13. prototype[property] = function(...args) {
  14. const self = this.getInnerGuides();
  15. if (!self || !self[property]) {
  16. return;
  17. }
  18. return self[property](...args);
  19. };
  20. })
  21. @Properties(PROPERTIES, (prototype, property) => {
  22. Object.defineProperty(prototype, property, {
  23. get() {
  24. return this.getInnerGuides().props[property];
  25. },
  26. set(value) {
  27. this.innerGuides.setState({
  28. [property]: value,
  29. });
  30. },
  31. enumerable: true,
  32. configurable: true,
  33. });
  34. })
  35. /**
  36. * @sort 1
  37. * @extends EventEmitter
  38. */
  39. class Guides extends EventEmitter<GuidesEvents> {
  40. private containerProvider: ContainerProvider | null = null;
  41. private selfElement: HTMLElement | null = null;
  42. private _warp = false;
  43. private innerGuides!: InnerGuides;
  44. /**
  45. * @sort 1
  46. * @param - guides' container
  47. * @param {$ts:Partial<Guides.GuidesOptions>} - guides' options
  48. */
  49. constructor(container: HTMLElement, options: Partial<GuidesOptions> = {}) {
  50. super();
  51. const events: any = {};
  52. EVENTS.forEach(name => {
  53. events[camelize(`on ${name}`)] = (e: any) => this.trigger(name as any, e);
  54. });
  55. let selfElement!: HTMLElement;
  56. if (options.warpSelf) {
  57. delete options.warpSelf;
  58. this._warp = true;
  59. selfElement = container;
  60. } else {
  61. selfElement = document.createElement("div");
  62. container.appendChild(selfElement);
  63. }
  64. this.containerProvider = renderSelf(
  65. <InnerGuides ref={ref(this, "innerGuides")}
  66. {...events}
  67. {...options} />,
  68. selfElement,
  69. );
  70. }
  71. /**
  72. * @param state
  73. * @param callback
  74. */
  75. public setState(state: Partial<GuidesOptions>, callback?: () => void) {
  76. this.innerGuides.setState(state, callback);
  77. }
  78. /**
  79. * @param callback
  80. */
  81. public forceUpdate(callback?: () => void) {
  82. this.innerGuides.forceUpdate(callback);
  83. }
  84. /**
  85. * destroy guides
  86. */
  87. public destroy() {
  88. const selfElement = this.selfElement!;
  89. renderSelf(
  90. null,
  91. selfElement!,
  92. this.containerProvider,
  93. );
  94. if (!this._warp) {
  95. selfElement?.parentElement?.removeChild(selfElement);
  96. }
  97. this.selfElement = null;
  98. this.innerGuides = null;
  99. }
  100. private getInnerGuides() {
  101. return this.innerGuides.guides;
  102. }
  103. }
  104. interface Guides extends GuidesInterface {}
  105. export default Guides;