{-
	Copyright (C) 2018 Dr. Alistair Ward

	This file is part of BishBosh.

	BishBosh 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 3 of the License, or
	(at your option) any later version.

	BishBosh 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 BishBosh.  If not, see <http://www.gnu.org/licenses/>.
-}
{- |
 [@AUTHOR@]	Dr. Alistair Ward

 [@DESCRIPTION@]	Defines the physical colour used to represent each component of the display.
-}

module BishBosh.Attribute.ColourScheme (
-- * Types
-- ** Data-types
	ColourScheme(
--		MkColourScheme,
		getDarkPieceColour,
		getLightPieceColour,
		getDarkSquareColour,
		getLightSquareColour
	),
-- * Constants
	tag,
--	darkSquareColourTag,
--	lightSquareColourTag,
--	darkPieceColourTag,
--	lightPieceColourTag,
-- * Functions
-- ** Constructor
--	mkColourScheme
) where

import qualified	BishBosh.Attribute.PhysicalColour	as Attribute.PhysicalColour
import qualified	BishBosh.Data.Exception			as Data.Exception
import qualified	BishBosh.Text.ShowList			as Text.ShowList
import qualified	Control.Arrow
import qualified	Control.DeepSeq
import qualified	Control.Exception
import qualified	Data.Default
import qualified	Text.XML.HXT.Arrow.Pickle		as HXT

-- | Used to qualify XML.
tag :: String
tag	= "colourScheme"

darkPieceColourTag, darkSquareColourTag, lightPieceColourTag, lightSquareColourTag :: String
[darkPieceColourTag, darkSquareColourTag, lightPieceColourTag, lightSquareColourTag]	= [showString brightness $ showString component "Colour" | brightness <- ["dark", "light"], component <- ["Piece", "Square"]] -- List-comprehension.

-- | Defines the command-line options.
data ColourScheme	= MkColourScheme {
	getDarkPieceColour	:: Attribute.PhysicalColour.PhysicalColour,	-- ^ The physical colour of the dark pieces.
	getLightPieceColour	:: Attribute.PhysicalColour.PhysicalColour,	-- ^ The physical colour of the light pieces.
	getDarkSquareColour	:: Attribute.PhysicalColour.PhysicalColour,	-- ^ The physical colour of the dark squares of the board.
	getLightSquareColour	:: Attribute.PhysicalColour.PhysicalColour	-- ^ The physical colour of the light squares of the board.
} deriving Eq

instance Control.DeepSeq.NFData ColourScheme where
	rnf MkColourScheme {
		getDarkPieceColour	= darkPieceColour,
		getLightPieceColour	= lightPieceColour,
		getDarkSquareColour	= darkSquareColour,
		getLightSquareColour	= lightSquareColour
	} = Control.DeepSeq.rnf (
		darkPieceColour,
		lightPieceColour,
		darkSquareColour,
		lightSquareColour
	 )

instance Show ColourScheme where
	showsPrec _ MkColourScheme {
		getDarkPieceColour	= darkPieceColour,
		getLightPieceColour	= lightPieceColour,
		getDarkSquareColour	= darkSquareColour,
		getLightSquareColour	= lightSquareColour
	} = Text.ShowList.showsAssociationList' $ map (Control.Arrow.second shows) [
		(
			darkPieceColourTag,
			darkPieceColour
		), (
			lightPieceColourTag,
			lightPieceColour
		), (
			darkSquareColourTag,
			darkSquareColour
		), (
			lightSquareColourTag,
			lightSquareColour
		)
	 ]

instance Data.Default.Default ColourScheme where
	def = MkColourScheme {
		getDarkPieceColour	= Attribute.PhysicalColour.blue,
		getLightPieceColour	= Attribute.PhysicalColour.yellow,
		getDarkSquareColour	= Attribute.PhysicalColour.black,
		getLightSquareColour	= Attribute.PhysicalColour.white
	}

instance HXT.XmlPickler ColourScheme where
	xpickle	= HXT.xpElem tag . HXT.xpWrap (
		\(a, b, c, d) -> mkColourScheme a b c d,	-- Construct.
		\MkColourScheme {
			getDarkPieceColour	= darkPieceColour,
			getLightPieceColour	= lightPieceColour,
			getDarkSquareColour	= darkSquareColour,
			getLightSquareColour	= lightSquareColour
		} -> (
			darkPieceColour,
			lightPieceColour,
			darkSquareColour,
			lightSquareColour
		) -- Deconstruct.
	 ) $ HXT.xp4Tuple (
		getDarkPieceColour def `HXT.xpDefault` HXT.xpAttr darkPieceColourTag HXT.xpickle
	 ) (
		getLightPieceColour def `HXT.xpDefault` HXT.xpAttr lightPieceColourTag HXT.xpickle
	 ) (
		getDarkSquareColour def `HXT.xpDefault` HXT.xpAttr darkSquareColourTag HXT.xpickle
	 ) (
		getLightSquareColour def `HXT.xpDefault` HXT.xpAttr lightSquareColourTag HXT.xpickle
	 ) where
		def :: ColourScheme
		def	= Data.Default.def

-- | Smart constructor.
mkColourScheme
	:: Attribute.PhysicalColour.PhysicalColour	-- ^ Dark piece.
	-> Attribute.PhysicalColour.PhysicalColour	-- ^ Light piece.
	-> Attribute.PhysicalColour.PhysicalColour	-- ^ Dark square.
	-> Attribute.PhysicalColour.PhysicalColour	-- ^ Light square.
	-> ColourScheme
mkColourScheme darkPieceColour lightPieceColour darkSquareColour lightSquareColour
	| darkPieceColour `elem` bgColours		= Control.Exception.throw . Data.Exception.mkIncompatibleData . showString "BishBosh.Attribute.ColourScheme.mkColourScheme:\t" . showString darkPieceColourTag . Text.ShowList.showsAssociation . shows darkPieceColour . showString " must differ from the physical colour of both squares; " $ shows bgColours "."
	| lightPieceColour `elem` bgColours		= Control.Exception.throw . Data.Exception.mkIncompatibleData . showString "BishBosh.Attribute.ColourScheme.mkColourScheme:\t" . showString lightPieceColourTag . Text.ShowList.showsAssociation . shows lightPieceColour . showString " must differ from the physical colour of both squares; " $ shows bgColours "."
	| darkSquareColour == lightSquareColour		= Control.Exception.throw . Data.Exception.mkIncompatibleData . showString "BishBosh.Attribute.ColourScheme.mkColourScheme:\tthe physical colours of " . shows lightSquareColourTag . showString " & " $ shows darkSquareColourTag ", must differ."
	| otherwise					= MkColourScheme {
		getDarkPieceColour	= darkPieceColour,
		getLightPieceColour	= lightPieceColour,
		getDarkSquareColour	= darkSquareColour,
		getLightSquareColour	= lightSquareColour
	}
	where
		bgColours	= [darkSquareColour, lightSquareColour]