jsactions/core/Component.js

/** 
 * @license
 * Copyright (c) 2023 Gaurang Lade
 * 
 * MIT License
 * 
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */

import EventDispatcher from "../../createjs/EventDispatcher";
import EventUtils from '../utils/EventUtils';
import ElementUtils from '../utils/ElementUtils';
import ElementState from '../utils/ElementState';


/**
 *
 * The Component class is the base class for all visual components. 
 * 
 * When setting "_createDOMElement=false" while creating instance of Component, Component class skip render DOM element part, assuming Component DOM Element already available and it take DOM element reference by componentId during Component creation.  
 *
 *
 * Component HTML DOM wrapper:
        < div class="vjs-component ${componentId}" >
 *
 * Component Class provides lifecycle methods as follows to manage DOM Elements and Standard Form Controls.
 *
 * 1) Creation & Initialization : constructor, init, attach
 *
 * 2) Update : refresh
 *
 * 3) Destruction : detach, destroy
 *

 * Create custom component by extending base Component Class and overrides and implement following methods which will executes in below given order
 *

 * 1) Creation & Initialization : constructor, init, initComponent, createDOMContent, addEventHandler,  bind

 *
 * 2) Update : changeComponentState, refresh, setComponentRequired, setComponentReadOnly, setComponentEnabled
 *
 * 3) Destruction : removeEventHandler, unBind, destroy
 *
 */
class Component extends EventDispatcher {
    
    /**
     * Creates an instance of Component.
     * @example Create an instance of Component in View Class in "createViewContent" method
    
        createViewContent() {
        let tmpViewContentEl = this.createViewHTML();
        this.addToViewElement(tmpViewContentEl);

        this.cmpButton1 = new MyButton("cmpBtn",this.id,"helloContainer",true);
        this.cmpButton1.init("My Component");
        this.cmpButton1.attach();
        this.cmpButton1.enabled = true;
        }

     *
     * @param {string} [_id=null] - ComponentID
     * @param {string} [_parentViewId=null] - Component Parent View ID
     * @param {string} [_parentContainerId=null] - Component Parent Container ID
     * @param {boolean} [_createDOMElement=true] - True if Component DOM element create from Template , False if taken from DOM.
     * @memberof Component
     */
    constructor(_id = null, _parentViewId = null,_parentContainerId=null, _createDOMElement = true) {
        super();
        this.isEnabled= true;
        this.isAttached = false;
        this.elState = ElementState.DEFAULT;
        this.isRequired = false;
        this.isReadOnly = false;
        this.createDOMElement = _createDOMElement;
        this.id = _id;
        this.formId = "defaultform";
        if((_id == null)|| (_id == ""))
            this.id = ElementUtils.generateComponentId();
        this.parentViewId = _parentViewId;
        this.parentContainerId = _parentContainerId;
    }

    /**
     *
     * Initialise Component properties and model
     * Call by View or Parent
     * @memberof Component
     */
    init() {
		this.initComponent();
	}
    
    /**
     * @override
     * @description call by init method
     * @memberof Component
     */
    initComponent(){}
 
    /**
     * creates DOM Contents of Component
     * Overrides by SubClass
     * Call by attach Method
     */
    createDOMContent() {}

    /**
     *
     * Add Component Content DOM Element to Component
     * @param {string} _tmpCompContentEl - DOMElement 
     * @memberof Component
     */
    addToComponentElement(_tmpCompContentEl){
        let tmpCompElement = this.componentElement;
        tmpCompElement.insertAdjacentHTML('beforeend', _tmpCompContentEl);
    }

    /**
     *
     * 
     * @description Add Event Handlers for Component DOM Elements and Model
     * Call by attach & set domElement Method
     * @override
     * @memberof Component
     */
    addEventHandler() {}

    /**
     * @override
     * @description Remove Event Handlers for Component DOM Elements and Model
     * call by destroy method
     * @memberof Component
     */
    removeEventHandler() {}

    /**
     * Overrides by SubClass
     * Bind Component Properties with Model or other Components
     * call by attach & view
     */ 
    bind() {}

    /**
     * Overrides by SubClass
     * Remove Bind on Component Properties with Model or other Components
     */
    unBind() {}

    /**
     * returns Component DOM Element
     */
    get componentElement() {
        let tmpComponentEl = null;
        if (this.isAttached) {
            tmpComponentEl = ElementUtils.component(this.id,this.parentViewId);
        }
        return tmpComponentEl;
    }
    
    /**
     * returns ParentViewID
     */
    get parentView(){
        return this.parentViewId;
    }

    /**
     * returns Parent Container ID
     */
    get parentContainer(){
        return this.parentContainerId;
    }

    /**
     * returns Component current state
     */
    get currentState() {
        return this.elState;
    }

    /**
     * change Component State
     */
    changeState(_state = "default"){
        this.elState = _state;
        this.changeComponentState();
    }

    /**
     * change Component State
     * Overrides by Subclass
     */
    changeComponentState(){}

    /**
     * returns True if Component is Required
     */
    get required(){
        return this.isRequired;
    }

    set required(_isRequired = false){
        this.isRequired = _isRequired;
        this.setComponentRequired();
    }

    /**
     * Overrides by Subclass
     */
    setComponentRequired(){}

    /**
     * returns True if Component is ResdyOnly
     */
    get readOnly(){
        return this.isReadOnly;
    }

    set readOnly(_isReadOnly = false){
        this.isReadOnly = _isReadOnly;
        if(this.isReadOnly == true){
            this.changeState(ElementState.READONLY);
        }else{
            this.changeState(ElementState.DEFAULT);
        }
        this.setComponentReadOnly();
    }  
    
    /**
     * Overrides by Subclass
     */
    setComponentReadOnly(){}


    /**
     * Boolean Property , returns True if Component is Enabled
     */
    get enabled() {
        return this.isEnabled;
    }

    set enabled(_isenabled = false){
        this.isEnabled = _isenabled;
        this.setComponentEnabled();
    }

    /**
     * Overrides by Subclass
     */
    setComponentEnabled(){}

    /**
     * Boolean Property , returns True if Component is Visible
     */
    get visible(){
        let tmpComponentEl = this.componentElement;
        return ElementUtils.isVisible(tmpComponentEl);
    }

    set visible(_visible){
        let tmpComponentEl = this.componentElement;
        if(_visible)
            ElementUtils.showElement(tmpComponentEl);
        else 
            ElementUtils.hideElement(tmpComponentEl);

    }

     /**
     * Component Lifecycle Method
     * Call by View or call manually 
     * attach Responsible to render component content, listners etc
     * Dispatch "ATTACHED_EVENT" when view got attached / rendered 
     * Following methods call by attach method
     * createDOMContent
     * addViewHandler
     * bind
     *  
     */
     attach() {
        if (!this.isAttached) {   
            this.isAttached = true;
            if(this.createDOMElement == true){
                let parentEl = ElementUtils.container(this.parentContainerId,this.parentViewId);
                parentEl.insertAdjacentHTML('beforeend', ElementUtils.constructComponentBaseElement(this.id));
                let tmpEle = this.componentElement;
                ElementUtils.hideElement(tmpEle);
                //Create  Contents and Internal Components
                this.createDOMContent();
            }
            
            //Add Event Handlers for view Internal Components
            this.addEventHandler();
            //Bind  Component Properties with Model or other  Components
            this.bind();
            this.enabled = true;
            this.visible = true;
            this.dispatchEvent(EventUtils.ATTACHED_EVENT);
        }
    }

    /**
     * Component Lifecycle Method
     * Call by View or call manually 
     * Component will be remove if Component is attached
    */
    detach() {
        if (this.isAttached) {
            this.unBind();
            this.removeEventHandler();
            let tmpComponentEle = this.componentElement;
            tmpComponentEle.parentNode.removeChild(tmpComponentEle);
            this.isAttached = false;
            this.dispatchEvent(EventUtils.DETACHED_EVENT);
        }
    }

    /**
     * Refresh Component data and model and event listners
     * Overrides by SubClass
     */
    refresh() {}

    /**
     * Overrides by SubClass
     * Destroy Method used to cleanup component resources
     * Call by View or call manually to destroy view
     * Remove Event Handlers, Make Properties null, 
     * Remove DOM Element contents and its reference.
     **/
    destroy() {}

}

export default Component;