module Gargantext.Pages.Corpus.Chart.Pie where import Data.String (take, joinWith, Pattern(..), split, length) import Data.Array (foldl, zip, filter) import Data.Array as A import Data.Tuple (Tuple(..)) import Data.Map as Map import Data.Int (toNumber) import Data.Map (Map) import Data.Argonaut (class DecodeJson, decodeJson, (.?)) import Data.Maybe (Maybe(..), maybe) import Effect.Aff (Aff) import Gargantext.Config -- (End(..), Path(..), TabType, toUrl) import Gargantext.Config.REST (get) import React (ReactClass, ReactElement, createElement) import Thermite (Spec, Render, defaultPerformAction, simpleSpec, createClass) import Gargantext.Prelude import Gargantext.Types (TermList(..)) import Gargantext.Components.Loader as Loader import Gargantext.Components.Charts.Options.ECharts import Gargantext.Components.Charts.Options.Type import Gargantext.Components.Charts.Options.Series import Gargantext.Components.Charts.Options.Color import Gargantext.Components.Charts.Options.Font import Gargantext.Components.Charts.Options.Data type Path = { corpusId :: Int , tabType :: TabType } newtype ChartMetrics = ChartMetrics { "data" :: HistoMetrics } instance decodeChartMetrics :: DecodeJson ChartMetrics where decodeJson json = do obj <- decodeJson json d <- obj .? "data" pure $ ChartMetrics { "data": d } newtype HistoMetrics = HistoMetrics { dates :: Array String , count :: Array Number } instance decodeHistoMetrics :: DecodeJson HistoMetrics where decodeJson json = do obj <- decodeJson json d <- obj .? "dates" c <- obj .? "count" pure $ HistoMetrics { dates : d , count: c} type Loaded = HistoMetrics loadedMetricsSpec :: Spec {} (Loader.InnerProps Path Loaded ()) Void loadedMetricsSpec = simpleSpec defaultPerformAction render where render :: Render {} (Loader.InnerProps Path Loaded ()) Void render dispatch {loaded : metricsData} {} _ = [chart (chartOptions metricsData)] chartOptions :: HistoMetrics -> Options chartOptions (HistoMetrics { dates: dates', count: count'}) = Options { mainTitle : "Bar" , subTitle : "Count of GraphTerm" , xAxis : xAxis' $ map (\t -> joinWith " " $ map (take 3) $ A.take 3 $ filter (\s -> length s > 3) $ split (Pattern " ") t) dates' , yAxis : yAxis' { position: "left", show: true, min:0} , series : [seriesBarD1 {name: "Number of publication / year"} $ map (\n -> dataSerie {name: "", itemStyle: itemStyle {color:blue}, value: n }) count'] , addZoom : false , tooltip : mkTooltip { formatter: templateFormatter "{b0}" } } loadedMetricsSpecPie :: Spec {} (Loader.InnerProps Path Loaded ()) Void loadedMetricsSpecPie = simpleSpec defaultPerformAction render where render :: Render {} (Loader.InnerProps Path Loaded ()) Void render dispatch {loaded : metricsData} {} _ = [chart (chartOptionsPie metricsData)] chartOptionsPie :: HistoMetrics -> Options chartOptionsPie (HistoMetrics { dates: dates', count: count'}) = Options { mainTitle : "Pie" , subTitle : "Distribution by GraphTerm" , xAxis : xAxis' [] , yAxis : yAxis' { position: "", show: false, min:0} , series : [seriesPieD1 {name: "Data"} $ map (\(Tuple n v) -> dataSerie {name: n, value:v}) $ zip dates' count'] -- , series : [seriesBarD1 {name: "Number of publication / year"} $ map (\n -> dataSerie {name: "", value: n }) count'] , addZoom : false , tooltip : mkTooltip { formatter: templateFormatter "{b0}" } } metricsLoader :: Loader.Props' Path HistoMetrics -> ReactElement metricsLoader props = createElement metricsLoaderClass props [] where metricsLoaderClass :: ReactClass (Loader.Props Path HistoMetrics) metricsLoaderClass = Loader.createLoaderClass "MetricsLoader" getMetrics getMetrics :: Path -> Aff HistoMetrics getMetrics {corpusId, tabType:tabType} = do ChartMetrics ms <- get $ toUrl Back (Chart {chartType: ChartPie, tabType: tabType}) $ Just corpusId pure ms."data" pieSpec :: Spec {} Path Void pieSpec = simpleSpec defaultPerformAction render where render :: Render {} Path Void render dispatch path {} _ = [ metricsLoader { path , component: createClass "LoadedMetrics" loadedMetricsSpecPie (const {}) } ] barSpec :: Spec {} Path Void barSpec = simpleSpec defaultPerformAction render where render :: Render {} Path Void render dispatch path {} _ = [ metricsLoader { path , component: createClass "LoadedMetrics" loadedMetricsSpec (const {}) } ]