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. function isUndefined(value: any): boolean {
  6. return typeof value === "undefined";
  7. }
  8. interface DefaultProps<T> {
  9. eventType: string;
  10. stop: () => void;
  11. currentTarget: T;
  12. }
  13. type NotFunction = { [k: string]: unknown } & ({ bind?: never } | { call?: never });
  14. type NoArguments = undefined | null | void | never;
  15. type EventWithRestParam = ((evt: NotFunction, ...restParam: any[]) => any);
  16. /**
  17. * Types that can be used when attaching new event definition as generic on a class
  18. * @ko 클래스 타입 등록시 사용가능한 타입
  19. * @example
  20. * ```
  21. * new SomeClass<{
  22. * // Using it as object
  23. * evt0: {
  24. * param0: number;
  25. * param1: string
  26. * };
  27. * // Using it as function with other arguments
  28. * evt1: (arg0: {
  29. * param0: number;
  30. * param1: string
  31. * }, arg1: string, arg2: boolean) => boolean;
  32. * }>
  33. * ```
  34. */
  35. type EventDefinition = NotFunction | NoArguments | EventWithRestParam;
  36. type EventMap = Record<string, EventDefinition>;
  37. type EventKey<T extends EventMap> = string & keyof T;
  38. type EventHash<T extends EventMap, S> = Partial<{ [K in EventKey<T>]: EventCallback<T, K, S> }>;
  39. type EventCallbackFirstParam<P, S> = P extends NoArguments ? DefaultProps<S> : P & DefaultProps<S>;
  40. type EventCallbackFunction<T extends (...params: any[]) => any, S>
  41. = T extends (firstParam?: infer F, ...restParams: infer R) => any
  42. ? (firstParam: EventCallbackFirstParam<Required<F>, S>, ...restParams: R) => any
  43. : (firstParam: DefaultProps<S>) => any;
  44. // In the on and once methods, the defaultProps must be included in the first parameter.
  45. type EventCallback<T extends EventMap, K extends EventKey<T>, S>
  46. = T[K] extends (...params: any[]) => any
  47. ? EventCallbackFunction<T[K], S>
  48. : (event: EventCallbackFirstParam<T[K], S>) => any;
  49. type EventTriggerFirstParam<T extends {}> = Pick<T, Exclude<keyof T, keyof DefaultProps<any>>> & Partial<DefaultProps<any>>;
  50. type EventDiff<T, U> = T extends U ? never : T;
  51. type EventTriggerPartialFunction<T extends (...params: any[]) => any>
  52. = T extends (firstParam: infer F, ...restParam: infer R) => any
  53. ? (firstParam?: EventTriggerFirstParam<EventDiff<F, undefined>>, ...restParams: R) => any
  54. : never;
  55. type EventTriggerRequiredFunction<T extends (...params: any[]) => any>
  56. = T extends (firstParam: infer F, ...restParam: infer R) => any
  57. ? (firstParam: EventTriggerFirstParam<F>, ...restParams: R) => any
  58. : never;
  59. type EventTriggerFunction<T extends (...params: any[]) => any>
  60. = Parameters<T> extends Required<Parameters<T>> & [any]
  61. ? EventTriggerRequiredFunction<T>
  62. : EventTriggerPartialFunction<T>
  63. type EventTriggerNoFunction<T>
  64. = T extends NoArguments
  65. ? (firstParam?: { [key: string]: never }) => any
  66. : EventTriggerFunction<(fisrtParam: EventTriggerFirstParam<T>) => any>;
  67. // You don't need to include defaultProps in the trigger method's first parameter.
  68. type EventTriggerParams<T extends EventMap, K extends EventKey<T>>
  69. = Parameters<T[K] extends (...params: any[]) => any
  70. ? EventTriggerFunction<T[K]>
  71. : EventTriggerNoFunction<T[K]>>;
  72. interface DefaultEventMap {
  73. [key: string]: (firstParam?: { [key: string]: any }, ...restParams: any[]) => any;
  74. }
  75. /**
  76. * A class used to manage events in a component
  77. * @ko 컴포넌트의 이벤트을 관리할 수 있게 하는 클래스
  78. * @alias eg.Component
  79. */
  80. class Component<T extends EventMap = DefaultEventMap> {
  81. /**
  82. * Version info string
  83. * @ko 버전정보 문자열
  84. * @name VERSION
  85. * @static
  86. * @example
  87. * eg.Component.VERSION; // ex) 2.0.0
  88. * @memberof eg.Component
  89. */
  90. public static VERSION: string = "#__VERSION__#";
  91. /**
  92. * @deprecated
  93. * @private
  94. */
  95. public options: { [key: string]: any } = {};
  96. private _eventHandler: { [keys: string]: EventCallback<T, EventKey<T>, Component<T>>[] };
  97. /**
  98. * @support {"ie": "7+", "ch" : "latest", "ff" : "latest", "sf" : "latest", "edge" : "latest", "ios" : "7+", "an" : "2.1+ (except 3.x)"}
  99. */
  100. constructor() {
  101. this._eventHandler = {};
  102. }
  103. public trigger<K extends EventKey<T>>(eventName: K, ...params: EventTriggerParams<T, K>): boolean;
  104. /**
  105. * Triggers a custom event.
  106. * @ko 커스텀 이벤트를 발생시킨다
  107. * @param {string} eventName The name of the custom event to be triggered <ko>발생할 커스텀 이벤트의 이름</ko>
  108. * @param {object} customEvent Event data to be sent when triggering a custom event <ko>커스텀 이벤트가 발생할 때 전달할 데이터</ko>
  109. * @param {any[]} restParam Additional parameters when triggering a custom event <ko>커스텀 이벤트가 발생할 때 필요시 추가적으로 전달할 데이터</ko>
  110. * @return Indicates whether the event has occurred. If the stop() method is called by a custom event handler, it will return false and prevent the event from occurring. <a href="https://github.com/naver/egjs-component/wiki/How-to-make-Component-event-design%3F">Ref</a> <ko>이벤트 발생 여부. 커스텀 이벤트 핸들러에서 stop() 메서드를 호출하면 'false'를 반환하고 이벤트 발생을 중단한다. <a href="https://github.com/naver/egjs-component/wiki/How-to-make-Component-event-design%3F">참고</a></ko>
  111. * @example
  112. * ```
  113. * class Some extends eg.Component {
  114. * some(){
  115. * if(this.trigger("beforeHi")){ // When event call to stop return false.
  116. * this.trigger("hi");// fire hi event.
  117. * }
  118. * }
  119. * }
  120. *
  121. * const some = new Some();
  122. * some.on("beforeHi", (e) => {
  123. * if(condition){
  124. * e.stop(); // When event call to stop, `hi` event not call.
  125. * }
  126. * });
  127. * some.on("hi", (e) => {
  128. * // `currentTarget` is component instance.
  129. * console.log(some === e.currentTarget); // true
  130. * });
  131. * // If you want to more know event design. You can see article.
  132. * // https://github.com/naver/egjs-component/wiki/How-to-make-Component-event-design%3F
  133. * ```
  134. */
  135. public trigger<K extends EventKey<T>>(eventName: K, ...params: any[]): boolean {
  136. let handlerList = this._eventHandler[eventName] || [];
  137. const hasHandlerList = handlerList.length > 0;
  138. if (!hasHandlerList) {
  139. return true;
  140. }
  141. const customEvent = params[0] || {};
  142. const restParams = params.slice(1);
  143. // If detach method call in handler in first time then handler list calls.
  144. handlerList = handlerList.concat();
  145. let isCanceled = false;
  146. // This should be done like this to pass previous tests
  147. (customEvent as any).eventType = eventName;
  148. (customEvent as any).stop = () => { isCanceled = true; };
  149. (customEvent as any).currentTarget = this;
  150. let arg: any[] = [customEvent];
  151. if (restParams.length >= 1) {
  152. arg = arg.concat(restParams);
  153. }
  154. handlerList.forEach(handler => {
  155. handler.apply(this, arg);
  156. });
  157. return !isCanceled;
  158. }
  159. public once<K extends EventKey<T>>(eventName: K, handlerToAttach: EventCallback<T, K, this>): this;
  160. public once(eventHash: EventHash<T, this>): this;
  161. /**
  162. * Executed event just one time.
  163. * @ko 이벤트가 한번만 실행된다.
  164. * @param {string} eventName The name of the event to be attached <ko>등록할 이벤트의 이름</ko>
  165. * @param {function} handlerToAttach The handler function of the event to be attached <ko>등록할 이벤트의 핸들러 함수</ko>
  166. * @return An instance of a component itself<ko>컴포넌트 자신의 인스턴스</ko>
  167. * @example
  168. * ```
  169. * class Some extends eg.Component {
  170. * hi() {
  171. * alert("hi");
  172. * }
  173. * thing() {
  174. * this.once("hi", this.hi);
  175. * }
  176. *
  177. * var some = new Some();
  178. * some.thing();
  179. * some.trigger("hi");
  180. * // fire alert("hi");
  181. * some.trigger("hi");
  182. * // Nothing happens
  183. * ```
  184. */
  185. public once<K extends EventKey<T>>(eventName: K | EventHash<T, this>, handlerToAttach?: EventCallback<T, K, this>): this {
  186. if (typeof eventName === "object" && isUndefined(handlerToAttach)) {
  187. const eventHash = eventName;
  188. for (const key in eventHash) {
  189. this.once((key as K), eventHash[key] as EventCallback<T, K, this>);
  190. }
  191. return this;
  192. } else if (typeof eventName === "string" && typeof handlerToAttach === "function") {
  193. const listener: any = (...args: any[]) => {
  194. handlerToAttach.apply(this, args);
  195. this.off(eventName, listener);
  196. }
  197. this.on(eventName, listener);
  198. }
  199. return this;
  200. }
  201. /**
  202. * Checks whether an event has been attached to a component.
  203. * @ko 컴포넌트에 이벤트가 등록됐는지 확인한다.
  204. * @param {string} eventName The name of the event to be attached <ko>등록 여부를 확인할 이벤트의 이름</ko>
  205. * @return {boolean} Indicates whether the event is attached. <ko>이벤트 등록 여부</ko>
  206. * @example
  207. * ```
  208. * class Some extends eg.Component {
  209. * some() {
  210. * this.hasOn("hi");// check hi event.
  211. * }
  212. * }
  213. * ```
  214. */
  215. public hasOn<K extends EventKey<T>>(eventName: K): boolean {
  216. return !!this._eventHandler[eventName];
  217. }
  218. public on<K extends EventKey<T>>(eventName: K, handlerToAttach: EventCallback<T, K, this>): this;
  219. public on(eventHash: EventHash<T, this>): this;
  220. /**
  221. * Attaches an event to a component.
  222. * @ko 컴포넌트에 이벤트를 등록한다.
  223. * @param {string} eventName The name of the event to be attached <ko>등록할 이벤트의 이름</ko>
  224. * @param {function} handlerToAttach The handler function of the event to be attached <ko>등록할 이벤트의 핸들러 함수</ko>
  225. * @return An instance of a component itself<ko>컴포넌트 자신의 인스턴스</ko>
  226. * @example
  227. * ```
  228. * class Some extends eg.Component {
  229. * hi() {
  230. * console.log("hi");
  231. * }
  232. * some() {
  233. * this.on("hi",this.hi); //attach event
  234. * }
  235. * }
  236. * ```
  237. */
  238. public on<K extends EventKey<T>>(eventName: K | EventHash<T, this>, handlerToAttach?: EventCallback<T, K, this>): this {
  239. if (typeof eventName === "object" && isUndefined(handlerToAttach)) {
  240. const eventHash = eventName;
  241. for (const name in eventHash) {
  242. this.on(name, eventHash[name] as any);
  243. }
  244. return this;
  245. } else if (typeof eventName === "string" &&
  246. typeof handlerToAttach === "function") {
  247. let handlerList = this._eventHandler[eventName];
  248. if (isUndefined(handlerList)) {
  249. this._eventHandler[eventName] = [];
  250. handlerList = this._eventHandler[eventName];
  251. }
  252. handlerList.push(handlerToAttach as EventCallback<T, EventKey<T>, this>);
  253. }
  254. return this;
  255. }
  256. public off(eventHash?: EventHash<T, this>): this;
  257. public off<K extends EventKey<T>>(eventName: K, handlerToDetach?: EventCallback<T, K, this>): this;
  258. /**
  259. * Detaches an event from the component.
  260. * @ko 컴포넌트에 등록된 이벤트를 해제한다
  261. * @param {string} eventName The name of the event to be detached <ko>해제할 이벤트의 이름</ko>
  262. * @param {function} handlerToDetach The handler function of the event to be detached <ko>해제할 이벤트의 핸들러 함수</ko>
  263. * @return An instance of a component itself <ko>컴포넌트 자신의 인스턴스</ko>
  264. * @example
  265. * ```
  266. * class Some extends eg.Component {
  267. * hi() {
  268. * console.log("hi");
  269. * }
  270. * some() {
  271. * this.off("hi",this.hi); //detach event
  272. * }
  273. * }
  274. * ```
  275. */
  276. public off<K extends EventKey<T>>(eventName?: K | EventHash<T, this>, handlerToDetach?: EventCallback<T, K, this>): this {
  277. // Detach all event handlers.
  278. if (isUndefined(eventName)) {
  279. this._eventHandler = {};
  280. return this;
  281. }
  282. // Detach all handlers for eventname or detach event handlers by object.
  283. if (isUndefined(handlerToDetach)) {
  284. if (typeof eventName === "string") {
  285. delete this._eventHandler[eventName];
  286. return this;
  287. } else {
  288. const eventHash = eventName;
  289. for (const name in eventHash) {
  290. this.off(name, eventHash[name] as any);
  291. }
  292. return this;
  293. }
  294. }
  295. // Detach single event handler
  296. const handlerList = this._eventHandler[eventName as K];
  297. if (handlerList) {
  298. let idx = 0;
  299. for (const handlerFunction of handlerList) {
  300. if (handlerFunction === handlerToDetach) {
  301. handlerList.splice(idx, 1);
  302. break;
  303. }
  304. idx++;
  305. }
  306. }
  307. return this;
  308. }
  309. }
  310. export default Component;
comments powered by Disqus