Source: node_modules/@egjs/component/src/Component.ts

  1. /*
  2. * Copyright (c) 2015 NAVER Corp.
  3. * egjs projects are licensed under the MIT license
  4. */
  5. import { isUndefined } from "./utils";
  6. import { EventCallback, EventHash, EventKey, EventMap, EventTriggerParams } from "./types";
  7. import ComponentEvent from "./ComponentEvent";
  8. import ActualComponentEvent from "./ActualComponentEvent";
  9. /**
  10. * A class used to manage events in a component
  11. * @ko 컴포넌트의 이벤트을 관리할 수 있게 하는 클래스
  12. */
  13. class Component<T extends EventMap> {
  14. /**
  15. * Version info string
  16. * @ko 버전정보 문자열
  17. * @name VERSION
  18. * @static
  19. * @example
  20. * Component.VERSION; // ex) 3.0.0
  21. * @memberof Component
  22. */
  23. public static VERSION: string = "#__VERSION__#";
  24. private _eventHandler: { [keys: string]: Array<(...args: any[]) => any> };
  25. /**
  26. * @support {"ie": "7+", "ch" : "latest", "ff" : "latest", "sf" : "latest", "edge" : "latest", "ios" : "7+", "an" : "2.1+ (except 3.x)"}
  27. */
  28. public constructor() {
  29. this._eventHandler = {};
  30. }
  31. public trigger<K extends EventKey<T>>(event: ComponentEvent<T[K], K, this> & T[K]): this;
  32. public trigger<K extends EventKey<T>>(event: K, ...params: EventTriggerParams<T, K>): this;
  33. /**
  34. * Trigger a custom event.
  35. * @ko 커스텀 이벤트를 발생시킨다
  36. * @param {string | ComponentEvent} event The name of the custom event to be triggered or an instance of the ComponentEvent<ko>발생할 커스텀 이벤트의 이름 또는 ComponentEvent의 인스턴스</ko>
  37. * @param {any[]} params Event data to be sent when triggering a custom event <ko>커스텀 이벤트가 발생할 때 전달할 데이터</ko>
  38. * @return An instance of the component itself<ko>컴포넌트 자신의 인스턴스</ko>
  39. * @example
  40. * ```ts
  41. * import Component, { ComponentEvent } from "@egjs/component";
  42. *
  43. * class Some extends Component<{
  44. * beforeHi: ComponentEvent<{ foo: number; bar: string }>;
  45. * hi: { foo: { a: number; b: boolean } };
  46. * someEvent: (foo: number, bar: string) => void;
  47. * someOtherEvent: void; // When there's no event argument
  48. * }> {
  49. * some(){
  50. * if(this.trigger("beforeHi")){ // When event call to stop return false.
  51. * this.trigger("hi");// fire hi event.
  52. * }
  53. * }
  54. * }
  55. *
  56. * const some = new Some();
  57. * some.on("beforeHi", e => {
  58. * if(condition){
  59. * e.stop(); // When event call to stop, `hi` event not call.
  60. * }
  61. * // `currentTarget` is component instance.
  62. * console.log(some === e.currentTarget); // true
  63. *
  64. * typeof e.foo; // number
  65. * typeof e.bar; // string
  66. * });
  67. * some.on("hi", e => {
  68. * typeof e.foo.b; // boolean
  69. * });
  70. * // If you want to more know event design. You can see article.
  71. * // https://github.com/naver/egjs-component/wiki/How-to-make-Component-event-design%3F
  72. * ```
  73. */
  74. public trigger<K extends EventKey<T>>(event: K | ComponentEvent<T[K], K, this>, ...params: EventTriggerParams<T, K> | void[]): this {
  75. const eventName = (event as any) instanceof ActualComponentEvent
  76. ? (event as ActualComponentEvent<T[K]>).eventType
  77. : event as K;
  78. const handlers = [...(this._eventHandler[eventName] || [])];
  79. if (handlers.length <= 0) {
  80. return this;
  81. }
  82. if ((event as any) instanceof ActualComponentEvent) {
  83. (event as ActualComponentEvent<T[K]>).currentTarget = this;
  84. handlers.forEach((handler: (event: ComponentEvent<T[K], K, this>) => any) => {
  85. handler(event as ComponentEvent<T[K], K, this>);
  86. });
  87. } else {
  88. handlers.forEach(handler => {
  89. // eslint-disable-next-line @typescript-eslint/no-unsafe-call
  90. handler(...params);
  91. });
  92. }
  93. return this;
  94. }
  95. public once<K extends EventKey<T>>(eventName: K, handlerToAttach: EventCallback<T, K, this>): this;
  96. public once(eventHash: EventHash<T, this>): this;
  97. /**
  98. * Executed event just one time.
  99. * @ko 이벤트가 한번만 실행된다.
  100. * @param {string} eventName The name of the event to be attached or an event name - event handler mapped object.<ko>등록할 이벤트의 이름 또는 이벤트 이름-핸들러 오브젝트</ko>
  101. * @param {function} handlerToAttach The handler function of the event to be attached <ko>등록할 이벤트의 핸들러 함수</ko>
  102. * @return An instance of the component itself<ko>컴포넌트 자신의 인스턴스</ko>
  103. * @example
  104. * ```ts
  105. * import Component, { ComponentEvent } from "@egjs/component";
  106. *
  107. * class Some extends Component<{
  108. * hi: ComponentEvent;
  109. * }> {
  110. * hi() {
  111. * alert("hi");
  112. * }
  113. * thing() {
  114. * this.once("hi", this.hi);
  115. * }
  116. * }
  117. *
  118. * var some = new Some();
  119. * some.thing();
  120. * some.trigger(new ComponentEvent("hi"));
  121. * // fire alert("hi");
  122. * some.trigger(new ComponentEvent("hi"));
  123. * // Nothing happens
  124. * ```
  125. */
  126. public once<K extends EventKey<T>>(eventName: K | EventHash<T, this>, handlerToAttach?: EventCallback<T, K, this>): this {
  127. if (typeof eventName === "object" && isUndefined(handlerToAttach)) {
  128. const eventHash = eventName;
  129. for (const key in eventHash) {
  130. this.once((key as K), eventHash[key] as EventCallback<T, K, this>);
  131. }
  132. return this;
  133. } else if (typeof eventName === "string" && typeof handlerToAttach === "function") {
  134. const listener: any = (...args: any[]) => {
  135. // eslint-disable-next-line @typescript-eslint/no-unsafe-call
  136. handlerToAttach(...args);
  137. this.off(eventName, listener);
  138. };
  139. this.on(eventName, listener);
  140. }
  141. return this;
  142. }
  143. /**
  144. * Checks whether an event has been attached to a component.
  145. * @ko 컴포넌트에 이벤트가 등록됐는지 확인한다.
  146. * @param {string} eventName The name of the event to be attached <ko>등록 여부를 확인할 이벤트의 이름</ko>
  147. * @return {boolean} Indicates whether the event is attached. <ko>이벤트 등록 여부</ko>
  148. * @example
  149. * ```ts
  150. * import Component from "@egjs/component";
  151. *
  152. * class Some extends Component<{
  153. * hi: void;
  154. * }> {
  155. * some() {
  156. * this.hasOn("hi");// check hi event.
  157. * }
  158. * }
  159. * ```
  160. */
  161. public hasOn<K extends EventKey<T>>(eventName: K): boolean {
  162. return !!this._eventHandler[eventName];
  163. }
  164. public on<K extends EventKey<T>>(eventName: K, handlerToAttach: EventCallback<T, K, this>): this;
  165. public on(eventHash: EventHash<T, this>): this;
  166. /**
  167. * Attaches an event to a component.
  168. * @ko 컴포넌트에 이벤트를 등록한다.
  169. * @param {string} eventName The name of the event to be attached or an event name - event handler mapped object.<ko>등록할 이벤트의 이름 또는 이벤트 이름-핸들러 오브젝트</ko>
  170. * @param {function} handlerToAttach The handler function of the event to be attached <ko>등록할 이벤트의 핸들러 함수</ko>
  171. * @return An instance of a component itself<ko>컴포넌트 자신의 인스턴스</ko>
  172. * @example
  173. * ```ts
  174. * import Component, { ComponentEvent } from "@egjs/component";
  175. *
  176. * class Some extends Component<{
  177. * hi: void;
  178. * }> {
  179. * hi() {
  180. * console.log("hi");
  181. * }
  182. * some() {
  183. * this.on("hi",this.hi); //attach event
  184. * }
  185. * }
  186. * ```
  187. */
  188. public on<K extends EventKey<T>>(eventName: K | EventHash<T, this>, handlerToAttach?: EventCallback<T, K, this>): this {
  189. if (typeof eventName === "object" && isUndefined(handlerToAttach)) {
  190. const eventHash = eventName;
  191. for (const name in eventHash) {
  192. this.on(name, eventHash[name] as any);
  193. }
  194. return this;
  195. } else if (typeof eventName === "string" &&
  196. typeof handlerToAttach === "function") {
  197. let handlerList = this._eventHandler[eventName];
  198. if (isUndefined(handlerList)) {
  199. this._eventHandler[eventName] = [];
  200. handlerList = this._eventHandler[eventName];
  201. }
  202. handlerList.push(handlerToAttach as EventCallback<T, EventKey<T>, this>);
  203. }
  204. return this;
  205. }
  206. public off(eventHash?: EventHash<T, this>): this;
  207. public off<K extends EventKey<T>>(eventName: K, handlerToDetach?: EventCallback<T, K, this>): this;
  208. /**
  209. * Detaches an event from the component.<br/>If the `eventName` is not given this will detach all event handlers attached.<br/>If the `handlerToDetach` is not given, this will detach all event handlers for `eventName`.
  210. * @ko 컴포넌트에 등록된 이벤트를 해제한다.<br/>`eventName`이 주어지지 않았을 경우 모든 이벤트 핸들러를 제거한다.<br/>`handlerToAttach`가 주어지지 않았을 경우 `eventName`에 해당하는 모든 이벤트 핸들러를 제거한다.
  211. * @param {string?} eventName The name of the event to be detached <ko>해제할 이벤트의 이름</ko>
  212. * @param {function?} handlerToDetach The handler function of the event to be detached <ko>해제할 이벤트의 핸들러 함수</ko>
  213. * @return An instance of a component itself <ko>컴포넌트 자신의 인스턴스</ko>
  214. * @example
  215. * ```ts
  216. * import Component, { ComponentEvent } from "@egjs/component";
  217. *
  218. * class Some extends Component<{
  219. * hi: void;
  220. * }> {
  221. * hi() {
  222. * console.log("hi");
  223. * }
  224. * some() {
  225. * this.off("hi",this.hi); //detach event
  226. * }
  227. * }
  228. * ```
  229. */
  230. public off<K extends EventKey<T>>(eventName?: K | EventHash<T, this>, handlerToDetach?: EventCallback<T, K, this>): this {
  231. // Detach all event handlers.
  232. if (isUndefined(eventName)) {
  233. this._eventHandler = {};
  234. return this;
  235. }
  236. // Detach all handlers for eventname or detach event handlers by object.
  237. if (isUndefined(handlerToDetach)) {
  238. if (typeof eventName === "string") {
  239. delete this._eventHandler[eventName];
  240. return this;
  241. } else {
  242. const eventHash = eventName;
  243. for (const name in eventHash) {
  244. this.off(name, eventHash[name] as any);
  245. }
  246. return this;
  247. }
  248. }
  249. // Detach single event handler
  250. const handlerList = this._eventHandler[eventName as K];
  251. if (handlerList) {
  252. let idx = 0;
  253. for (const handlerFunction of handlerList) {
  254. if (handlerFunction === handlerToDetach) {
  255. handlerList.splice(idx, 1);
  256. if (handlerList.length <= 0) {
  257. delete this._eventHandler[eventName as K];
  258. }
  259. break;
  260. }
  261. idx++;
  262. }
  263. }
  264. return this;
  265. }
  266. }
  267. export default Component;
comments powered by Disqus