/*
 * This program source code file is part of KiCad, a free EDA CAD application.
 *
 * Copyright (C) 2010 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
 * Copyright (C) 2010-2022 KiCad Developers, see change_log.txt for contributors.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, you may find one here:
 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
 * or you may search the http://www.gnu.org website for the version 2 license,
 * or you may write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
 */

#ifndef LAYERWIDGET_H_
#define LAYERWIDGET_H_

#include <wx/intl.h>
#include <wx/statbmp.h>
#include <wx/string.h>
#include <wx/aui/auibook.h>
#include <wx/notebook.h>
#include <wx/sizer.h>
#include <wx/gdicmn.h>
#include <wx/scrolwin.h>
#include <wx/font.h>
#include <wx/colour.h>
#include <wx/settings.h>
#include <wx/panel.h>
#include <wx/bitmap.h>
#include <wx/image.h>
#include <wx/icon.h>
#include <layer_ids.h>
#include <gal/color4d.h>
#include <widgets/color_swatch.h>
#include <widgets/indicator_icon.h>

#define LYR_COLUMN_COUNT        5           ///< Layer tab column count
#define RND_COLUMN_COUNT        2           ///< Rendering tab column count

#define COLUMN_ICON_ACTIVE 0
#define COLUMN_COLORBM 1
#define COLUMN_COLOR_LYR_CB 2
#define COLUMN_COLOR_LYRNAME 3
#define COLUMN_ALPHA_INDICATOR 4

using KIGFX::COLOR4D;

/**
 * Manage a list of layers with the notion of a "current" layer, and layer specific visibility
 * control.
 *
 * You must derive from it to use it so you can implement the abstract functions which receive
 * the events.  Each layer is given its own color, and that color can be changed within the UI
 * provided here.  This widget knows nothing of the client code, meaning it has no knowledge of
 * a BOARD or anything.  To use it you must derive from this class and implement the abstract
 * functions:
 * <p> void OnLayerColorChange( int aLayer, int aColor );
 * <p> bool OnLayerSelect( int aLayer );
 * <p> void OnLayerVisible( int aLayer, bool isVisible );
 * <p> void OnRenderColorChange( int id, int aColor );
 * <p> void OnRenderEnable( int id, bool isEnabled );
 *
 * @note Even if designed toward layers, it is used to contain other stuff, too (the second page
 * in pcbnew contains render items, for example).
 */
class LAYER_WIDGET : public wxPanel
{
public:
    /**
     * Provide all the data needed to add a row to a LAYER_WIDGET.  This is
     * part of the public API for a LAYER_WIDGET.
     */
    struct ROW
    {
        wxString    rowName;      ///< the prompt or layername
        int         id;           ///< either a layer or "visible element" id
        COLOR4D     color;        ///< COLOR4D::UNSPECIFIED if none.
        bool        state;        ///< initial wxCheckBox state
        wxString    tooltip;      ///< if not empty, use this tooltip on row
        bool        changeable;   ///< if true, the state can be changed
        bool        spacer;       ///< if true, this row is a spacer
        COLOR4D     defaultColor; ///< The default color for the row

        ROW( const wxString& aRowName, int aId, const COLOR4D& aColor = COLOR4D::UNSPECIFIED,
             const wxString& aTooltip = wxEmptyString, bool aState = true,
             bool aChangeable = true, const COLOR4D& aDefaultColor = COLOR4D::UNSPECIFIED )
        {
            rowName = aRowName;
            id      = aId;
            color   = aColor;
            state   = aState;
            tooltip = aTooltip;
            changeable = aChangeable;
            spacer = false;
            defaultColor = aDefaultColor;
        }

        ROW()
        {
            id = 0;
            color = COLOR4D::UNSPECIFIED;
            state = true;
            changeable = true;
            spacer = true;
            defaultColor = COLOR4D::UNSPECIFIED;
        }
    };

    static const wxEventType EVT_LAYER_COLOR_CHANGE;

public:

    /**
     * @param aParent is the parent window.
     * @param aFocusOwner is the window that should be sent the focus after.
     * @param id is the wxWindow id ( default = wxID_ANY).
     * @param pos is the window position.
     * @param size is the window size.
     * @param style is the window style.
     */
    LAYER_WIDGET( wxWindow* aParent, wxWindow* aFocusOwner, wxWindowID id = wxID_ANY,
                  const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxDefaultSize,
                  long style = wxTAB_TRAVERSAL );

    virtual ~LAYER_WIDGET();

    /**
     * Set the string that is used for determining the smallest string displayed in the layer's tab.
     */
    void SetSmallestLayerString( const wxString& aString )
    {
        m_smallestLayerString = aString;
    }

    /**
     * Return the preferred minimum size, taking into consideration the dynamic content.
     *
     * Nothing in wxWidgets was reliable enough so this overrides one of their functions.
     */
    wxSize GetBestSize() const;

    /**
     * Return the number of rows in the layer tab.
     */
    int GetLayerRowCount() const;

    /**
     * Return the number of rows in the render tab.
     */
    int GetRenderRowCount() const;

    /**
     * Append a new row in the layer portion of the widget.
     *
     * The user must ensure that ROW::id is unique for all existing rows on Windows.
     */
    void AppendLayerRow( const ROW& aRow );

    /**
     * Append new rows in the layer portion of the widget.
     *
     * The user must ensure that ROW::id is unique for all existing rows on Windows.
     */
    void AppendLayerRows( const ROW* aRowsArray, int aRowCount )
    {
        for( int row=0;  row<aRowCount;  ++row )
            AppendLayerRow( aRowsArray[row] );

        UpdateLayouts();
    }

    /**
     * Empty out the layer rows.
     */
    void ClearLayerRows();

    /**
     * Append a new row in the render portion of the widget.
     *
     * The user must ensure that ROW::id is unique for all existing rows on Windows.
     */
    void AppendRenderRow( const ROW& aRow );

    /**
     * Append new rows in the render portion of the widget.
     *
     * The user must ensure that ROW::id is unique for all existing rows on Windows.
     */
    void AppendRenderRows( const ROW* aRowsArray, int aRowCount )
    {
        for( int row=0;  row<aRowCount;  ++row )
            AppendRenderRow( aRowsArray[row] );

        UpdateLayouts();
    }

    /**
     * Empty out the render rows.
     */
    void ClearRenderRows();

    /**
     * Change the row selection in the layer list to the given row.
     */
    void SelectLayerRow( int aRow );

    /**
     * Change the row selection in the layer list to \a aLayer provided.
     */
    void SelectLayer( int aLayer );

    /**
     * Return the selected layer or -1 if none.
     */
    int GetSelectedLayer();

    /**
     * Set \a aLayer visible or not.  This does not invoke OnLayerVisible().
     */
    void SetLayerVisible( int aLayer, bool isVisible );

    /**
     * Return the visible state of the layer ROW associated with \a aLayer id.
     */
    bool IsLayerVisible( int aLayer );

    /**
     * Change the color of \a aLayer
     */
    void SetLayerColor( int aLayer, const COLOR4D& aColor );

    /**
     * Return the color of the layer ROW associated with \a aLayer id.
     */
    COLOR4D GetLayerColor( int aLayer ) const;

    /**
     * Return the color of the Render ROW in position \a aRow.
     */
    COLOR4D GetRenderColor( int aRow ) const;

    /**
     * Set the state of the checkbox associated with \a aId within the Render tab group of the
     * widget.
     *
     * This does not fire an event, i.e. does not invoke OnRenderEnable().
     *
     * @param aId is the same unique id used when adding a ROW to the Render tab.
     * @param isSet is the new checkbox state.
     */
    void SetRenderState( int aId, bool isSet );

    /**
     * Return the state of the checkbox associated with \a aId.
     *
     * @return true if checked, else false.
     */
    bool GetRenderState( int aId );

    void UpdateLayouts();

    /**
     * Update all layer manager icons (layers only).
     *
     * Useful when loading a file or clearing a layer because they change,
     * and the indicator arrow icon needs to be updated
     */
    void UpdateLayerIcons();

/*  did not help:
    void Freeze()
    {
        LAYER_PANEL_BASE::Freeze();
        m_LayerScrolledWindow->Freeze();
        m_RenderScrolledWindow->Freeze();
    }

    void Thaw()
    {
        m_RenderScrolledWindow->Thaw();
        m_LayerScrolledWindow->Thaw();
        LAYER_PANEL_BASE::Thaw();
    }
*/

    //-----<abstract functions>-------------------------------------------

    /**
     * Notify client code about a layer color change.
     *
     * Derived objects will handle this accordingly.
     *
     * @param aLayer is the board layer to change.
     * @param aColor is the new color.
     */
    virtual void OnLayerColorChange( int aLayer, const COLOR4D& aColor ) = 0;

    /**
     * Notify client code whenever the user selects a different layer.
     *
     * Derived classes will handle this accordingly, and can deny the change by returning false.
     *
     * @param aLayer is the board layer to select.
     */
    virtual bool OnLayerSelect( int aLayer ) = 0;

    /**
     * Notify client code about a layer visibility change.
     *
     * @param aLayer is the board layer to select.
     * @param isVisible is the new visible state.
     * @param isFinal is true when this is the last of potentially several such calls, and can
     *                be used to decide when to update the screen only one time instead of
     *                several times in the midst of a multiple layer change.
     */
    virtual void OnLayerVisible( int aLayer, bool isVisible, bool isFinal = true ) = 0;

    /**
     * Notify client code about a layer being right-clicked.
     *
     * @param aMenu is the right-click menu containing layer-scoped options.
     */
    virtual void OnLayerRightClick( wxMenu& aMenu ) = 0;

    /**
     * Notify client code whenever the user changes a rendering color.
     *
     * @param aId is the same id that was established in a Rendering row via the AddRenderRow()
     *            function.
     * @param aColor is the new color.
     */
    virtual void OnRenderColorChange( int aId, const COLOR4D& aColor ) = 0;

    /**
     * Notify client code whenever the user changes an rendering enable in one of the rendering
     * checkboxes.
     *
     * @param aId is the same id that was established in a Rendering row via the AddRenderRow()
     *            function.
     * @param isEnabled is the state of the checkbox, true if checked.
     */
    virtual void OnRenderEnable( int aId, bool isEnabled ) = 0;

protected:
    /**
     * @return true if bitmaps shown in Render layer list
     * are alternate bitmaps, or false if they are "normal" bitmaps
     * This is a virtual function because Pcbnew uses normal bitmaps
     * but GerbView uses both bitmaps
     * (alternate bitmaps to show layers in use, normal for others)
     */
    virtual bool useAlternateBitmap(int aRow) { return false; }

    /**
     * Subclasses can override this to provide accurate representation
     * of transparent color swatches.
     */
    virtual COLOR4D getBackgroundLayerColor() { return COLOR4D::BLACK; }

    /**
     * Allow saving a layer index within a control as its wxControl id.
     *
     * To do so in a way that all child wxControl ids within a wxWindow are unique, since this
     * is required by Windows.
     *
     * @see getDecodedId()
     */
    static int encodeId( int aColumn, int aId );

    /**
     * Decode \a aControlId to original un-encoded value.
     *
     * This holds if encodedId was called with a layer (this box is used for other things
     * than layers, too).
     */
    static int getDecodedId( int aControlId );

    void OnLeftDownLayers( wxMouseEvent& event );

    /**
     * Called when user right-clicks a layer.
     */
    void OnRightDownLayer( wxMouseEvent& event, COLOR_SWATCH* aColorSwatch,
                           const wxString& aLayerName );

    /**
     * Called when a user changes a swatch color.
     */
    void OnLayerSwatchChanged( wxCommandEvent& aEvent );

    /**
     * Handle the "is layer visible" checkbox and propagates the event to the client's
     * notification function.
     */
    void OnLayerCheckBox( wxCommandEvent& event );

    /**
     * Notify when user right-clicks a render option.
     */
    void OnRightDownRender( wxMouseEvent& aEvent, COLOR_SWATCH* aColorSwatch,
                            const wxString& aRenderName );

    /**
     * Called when user has changed the swatch color of a render entry.
     */
    void OnRenderSwatchChanged( wxCommandEvent& aEvent );

    void OnRenderCheckBox( wxCommandEvent& event );

    void OnTabChange( wxNotebookEvent& event );


    /**
     * Return the component within the m_LayersFlexGridSizer at @a aRow and @a aCol
     * or NULL if these parameters are out of range.
     *
     * @param aRow is the row index
     * @param aColumn is the column
     * @return the component installed within the sizer at given grid coordinate.
     */
    wxWindow* getLayerComp( int aRow, int aColumn ) const;
    wxWindow* getRenderComp( int aRow, int aColumn ) const;

    /**
     * Return the row index that \a aLayer resides in, or -1 if not found.
     */
    int findLayerRow( int aLayer ) const;
    int findRenderRow( int aId ) const;

    /**
     * Append or insert a new row in the layer portion of the widget.
     */
    void insertLayerRow( int aRow, const ROW& aSpec );

    void insertRenderRow( int aRow, const ROW& aSpec );

    void setLayerCheckbox( int aLayer, bool isVisible );

    void updateLayerRow( int aRow, const wxString& aName );

    /**
     * Give away the keyboard focus up to the main parent window.
     */
    void passOnFocus();

    // popup menu ids.
    enum POPUP_ID
    {
        ID_CHANGE_LAYER_COLOR = wxID_HIGHEST,
        ID_CHANGE_RENDER_COLOR,
        ID_LAST_VALUE
    };

    wxNotebook*         m_notebook;
    wxPanel*            m_LayerPanel;
    wxScrolledWindow*   m_LayerScrolledWindow;
    wxFlexGridSizer*    m_LayersFlexGridSizer;
    wxPanel*            m_RenderingPanel;
    wxScrolledWindow*   m_RenderScrolledWindow;
    wxFlexGridSizer*    m_RenderFlexGridSizer;

    wxWindow*           m_FocusOwner;
    int                 m_CurrentRow;           ///< selected row of layer list
    int                 m_PointSize;

    ROW_ICON_PROVIDER*  m_IconProvider;

    wxString            m_smallestLayerString;
};

#endif // LAYERWIDGET_H_