Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
P
purescript-gargantext
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
153
Issues
153
List
Board
Labels
Milestones
Merge Requests
3
Merge Requests
3
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
gargantext
purescript-gargantext
Commits
7a9fdad7
Commit
7a9fdad7
authored
Oct 25, 2021
by
arturo
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
>>> continue
parent
8e4975ab
Pipeline
#2016
failed with stage
Changes
6
Pipelines
1
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
595 additions
and
212 deletions
+595
-212
Phylo.purs
src/Gargantext/Components/Nodes/Corpus/Phylo.purs
+6
-5
Draw.js
src/Gargantext/Components/PhyloExplorer/Draw.js
+0
-0
JSON.purs
src/Gargantext/Components/PhyloExplorer/JSON.purs
+174
-0
Layout.purs
src/Gargantext/Components/PhyloExplorer/Layout.purs
+52
-40
Types.js
src/Gargantext/Components/PhyloExplorer/Types.js
+148
-0
Types.purs
src/Gargantext/Components/PhyloExplorer/Types.purs
+215
-167
No files found.
src/Gargantext/Components/Nodes/Corpus/Phylo.purs
View file @
7a9fdad7
...
@@ -12,8 +12,9 @@ import Data.HTTP.Method (Method(..))
...
@@ -12,8 +12,9 @@ import Data.HTTP.Method (Method(..))
import Data.Maybe (Maybe(..))
import Data.Maybe (Maybe(..))
import Effect.Aff (Aff, launchAff_)
import Effect.Aff (Aff, launchAff_)
import Effect.Class (liftEffect)
import Effect.Class (liftEffect)
import Gargantext.Components.PhyloExplorer.JSON (PhyloJSONSet)
import Gargantext.Components.PhyloExplorer.Layout (layout)
import Gargantext.Components.PhyloExplorer.Layout (layout)
import Gargantext.Components.PhyloExplorer.Types (PhyloData
s
et)
import Gargantext.Components.PhyloExplorer.Types (PhyloData
S
et)
import Gargantext.Sessions (Session)
import Gargantext.Sessions (Session)
import Gargantext.Types (NodeID)
import Gargantext.Types (NodeID)
import Gargantext.Utils.Reactix as R2
import Gargantext.Utils.Reactix as R2
...
@@ -36,7 +37,7 @@ phyloLayoutCpt :: R.Component Props
...
@@ -36,7 +37,7 @@ phyloLayoutCpt :: R.Component Props
phyloLayoutCpt = here.component "phyloLayout" cpt where
phyloLayoutCpt = here.component "phyloLayout" cpt where
cpt _ _ = do
cpt _ _ = do
fetchedDataBox <- T.useBox (Nothing :: Maybe Phylo
Datas
et)
fetchedDataBox <- T.useBox (Nothing :: Maybe Phylo
JSONS
et)
fetchedData <- T.useLive T.unequal fetchedDataBox
fetchedData <- T.useLive T.unequal fetchedDataBox
R.useEffectOnce' $ launchAff_ do
R.useEffectOnce' $ launchAff_ do
...
@@ -47,10 +48,10 @@ phyloLayoutCpt = here.component "phyloLayout" cpt where
...
@@ -47,10 +48,10 @@ phyloLayoutCpt = here.component "phyloLayout" cpt where
pure case fetchedData of
pure case fetchedData of
Nothing -> mempty
Nothing -> mempty
Just phyloData
set -> layout { phyloDatas
et } []
Just phyloData
Set -> layout { phyloDataS
et } []
fetchPhyloJSON :: Aff (Either String PhyloData
s
et)
fetchPhyloJSON :: Aff (Either String PhyloData
S
et)
fetchPhyloJSON =
fetchPhyloJSON =
let
let
request = AX.defaultRequest
request = AX.defaultRequest
...
@@ -65,4 +66,4 @@ fetchPhyloJSON =
...
@@ -65,4 +66,4 @@ fetchPhyloJSON =
Left err -> pure $ Left $ AX.printError err
Left err -> pure $ Left $ AX.printError err
Right response -> case JSON.readJSON response.body of
Right response -> case JSON.readJSON response.body of
Left err -> pure $ Left $ show err
Left err -> pure $ Left $ show err
Right (res :: Phylo
Dataset) -> pure $ Righ
t res
Right (res :: Phylo
JSONSet) -> pure $ Right $ parsePhyloJSONSe
t res
src/Gargantext/Components/PhyloExplorer/
d
raw.js
→
src/Gargantext/Components/PhyloExplorer/
D
raw.js
View file @
7a9fdad7
File moved
src/Gargantext/Components/PhyloExplorer/JSON.purs
0 → 100644
View file @
7a9fdad7
module Gargantext.Components.PhyloExplorer.JSON where
import Gargantext.Prelude
import Data.Generic.Rep (class Generic)
import Data.Generic.Rep as GR
import Data.Maybe (Maybe)
import Data.Show.Generic (genericShow)
import Gargantext.Utils.SimpleJSON (untaggedSumRep)
import Simple.JSON as JSON
type GraphData =
( bb :: String
, color :: String
, fontsize :: String
, label :: String
, labelloc :: String
, lheight :: String
, lp :: String
, lwidth :: String
, name :: String
, nodesep :: String
, overlap :: String
, phyloBranches :: String
, phyloDocs :: String
, phyloFoundations :: String
, phyloGroups :: String
, phyloPeriods :: String
, phyloSources :: String
, phyloTerms :: String
, phyloTimeScale :: String
, rank :: String
, ranksep :: String
, ratio :: String
, splines :: String
, style :: String
)
--------------------------------------------------
newtype PhyloJSONSet = PhyloJSONSet
{ _subgraph_cnt :: Int
, directed :: Boolean
, edges :: Array Edge
, objects :: Array PhyloObject
, strict :: Boolean
| GraphData
}
derive instance Generic PhyloJSONSet _
derive instance Eq PhyloJSONSet
instance Show PhyloJSONSet where show = genericShow
derive newtype instance JSON.ReadForeign PhyloJSONSet
--------------------------------------------------
type NodeData =
( height :: String
, label :: String
, name :: String
, nodeType :: String
, pos :: String
, shape :: String
, width :: String
)
data PhyloObject
= Layer
{ _gvid :: Int
, nodes :: Array Int
| GraphData
}
| BranchToNode
{ _gvid :: Int
, age :: String
, bId :: String
, birth :: String
, branchId :: String
, branch_x :: String
, branch_y :: String
, fillcolor :: String
, fontname :: String
, fontsize :: String
, size :: String
, style :: String
| NodeData
}
| GroupToNode
{ _gvid :: Int
, bId :: String
, branchId :: String
, fontname :: String
, foundation :: String
, frequence :: String
, from :: String
, lbl :: String
, penwidth :: String
, role :: String
, seaLvl :: String
, source :: String
, strFrom :: Maybe String
, strTo :: Maybe String
, support :: String
, to :: String
, weight :: String
| NodeData
}
| PeriodToNode
{ _gvid :: Int
, fontsize :: String
, from :: String
, strFrom :: Maybe String
, strTo :: Maybe String
, to :: String
| NodeData
}
derive instance Generic PhyloObject _
derive instance Eq PhyloObject
instance Show PhyloObject where show = genericShow
instance JSON.ReadForeign PhyloObject where
readImpl f = GR.to <$> untaggedSumRep f
--------------------------------------------------
type EdgeData =
( color :: String
, head :: Int
, pos :: String
, tail :: Int
, width :: String
)
data Edge
= GroupToGroup
{ _gvid :: Int
, constraint :: String
, edgeType :: String
, lbl :: String
, penwidth :: String
| EdgeData
}
| BranchToGroup
{ _gvid :: Int
, arrowhead :: String
, edgeType :: String
| EdgeData
}
| BranchToBranch
{ _gvid :: Int
, arrowhead :: String
, style :: String
| EdgeData
}
| GroupToAncestor
{ _gvid :: Int
, arrowhead :: String
, lbl :: String
, penwidth :: String
, style :: String
| EdgeData
}
| PeriodToPeriod
{ _gvid :: Int
| EdgeData
}
derive instance Generic Edge _
derive instance Eq Edge
instance Show Edge where show = genericShow
instance JSON.ReadForeign Edge where
readImpl f = GR.to <$> untaggedSumRep f
src/Gargantext/Components/PhyloExplorer/Layout.purs
View file @
7a9fdad7
...
@@ -4,12 +4,20 @@ module Gargantext.Components.PhyloExplorer.Layout
...
@@ -4,12 +4,20 @@ module Gargantext.Components.PhyloExplorer.Layout
import Gargantext.Prelude
import Gargantext.Prelude
import DOM.Simple
.Console (log2
)
import DOM.Simple
(Window, window
)
import Data.Array as Array
import Data.Array as Array
import Data.Int (fromString)
import Data.Date as Date
import Data.Maybe (maybe)
import Data.FoldableWithIndex (forWithIndex_)
import Data.Int as Int
import Data.Maybe (Maybe(..), maybe)
import Data.Number as Number
import Data.String as String
import Data.String as String
import Gargantext.Components.PhyloExplorer.Types (PhyloDataset(..))
import Data.Traversable (for)
import Data.Tuple as Tuple
import Data.Tuple.Nested ((/\))
import Effect (Effect)
import FFI.Simple (maybeGetProperty, (..), (...))
import Gargantext.Components.PhyloExplorer.Types (Group, PhyloDataSet(..))
import Gargantext.Utils (nbsp)
import Gargantext.Utils (nbsp)
import Gargantext.Utils.Reactix as R2
import Gargantext.Utils.Reactix as R2
import Reactix as R
import Reactix as R
...
@@ -20,40 +28,20 @@ here :: R2.Here
...
@@ -20,40 +28,20 @@ here :: R2.Here
here = R2.here "Gargantext.Components.PhyloExplorer"
here = R2.here "Gargantext.Components.PhyloExplorer"
type Props =
type Props =
( phyloData
set :: PhyloDatas
et
( phyloData
Set :: PhyloDataS
et
)
)
layout :: R2.Component Props
layout :: R2.Component Props
layout = R.createElement layoutCpt
layout = R.createElement layoutCpt
layoutCpt :: R.Component Props
layoutCpt :: R.Component Props
layoutCpt = here.component "layout" cpt where
layoutCpt = here.component "layout" cpt where
cpt { phyloData
set: (PhyloDataset phyloDataset
)
cpt { phyloData
Set: (PhyloDataSet o
)
} _ = do
} _ = do
-- States
-- States
let
{ phyloDocs
, phyloBranches
, phyloGroups
, phyloTerms
, phyloPeriods
, phyloFoundations
, phyloSources
} = phyloDataset
nbDocs = parseInt phyloDocs
nbBranches = parseInt phyloBranches
nbGroups = parseInt phyloGroups
nbTerms = parseInt phyloTerms
nbPeriods = parseInt phyloPeriods
nbFoundations = parseInt phyloFoundations
sourcesBox <- T.useBox (mempty :: Array String)
sources <- T.useLive T.unequal sourcesBox
-- Hooks
R.useEffectOnce' $ do
R.useEffectOnce' $ do
sources' <- pure $ stringArrToArr phyloSources
pure unit
T.write_ sources' sourcesBox
-- @hightlightSource
-- @hightlightSource
let
let
...
@@ -126,7 +114,7 @@ layoutCpt = here.component "layout" cpt where
...
@@ -126,7 +114,7 @@ layoutCpt = here.component "layout" cpt where
[ H.text "unselect source ✕" ]
[ H.text "unselect source ✕" ]
]
]
<>
<>
flip Array.mapWithIndex sources
flip Array.mapWithIndex
o.
sources
( \idx val ->
( \idx val ->
H.option
H.option
{ value: idx }
{ value: idx }
...
@@ -162,7 +150,10 @@ layoutCpt = here.component "layout" cpt where
...
@@ -162,7 +150,10 @@ layoutCpt = here.component "layout" cpt where
phyloCorpus {} []
phyloCorpus {} []
,
,
phyloCorpusInfo
phyloCorpusInfo
{ nbDocs, nbFoundations, nbPeriods }
{ nbDocs : o.nbDocs
, nbFoundations : o.nbFoundations
, nbPeriods : o.nbPeriods
}
[]
[]
,
,
-- H.div
-- H.div
...
@@ -176,7 +167,10 @@ layoutCpt = here.component "layout" cpt where
...
@@ -176,7 +167,10 @@ layoutCpt = here.component "layout" cpt where
phyloPhyloInfo
phyloPhyloInfo
{ nbTerms, nbGroups, nbBranches }
{ nbTerms : o.nbTerms
, nbGroups : o.nbGroups
, nbBranches : o.nbBranches
}
[]
[]
,
,
...
@@ -261,15 +255,33 @@ layoutCpt = here.component "layout" cpt where
...
@@ -261,15 +255,33 @@ layoutCpt = here.component "layout" cpt where
]
]
parseInt :: String -> Int
parseInt s = maybe 0 identity $ fromString s
stringArrToArr :: String -> Array String
stringArrToArr
setGlobalDependencies :: Window -> PhyloDataSet -> Effect Unit
= String.replace (String.Pattern "[") (String.Replacement "")
setGlobalDependencies w (PhyloDataSet o)
>>> String.replace (String.Pattern "]") (String.Replacement "")
= do
>>> String.split (String.Pattern ",")
-- _ <- w ... "freq" $ {}
>>> Array.filter (\s -> not eq 0 $ String.length s)
-- _ <- w ... "nbBranches" $ o.nbBranches
-- _ <- w ... "nbDocs" $ o.nbDocs
-- _ <- w ... "nbFoundations" $ o.nbFoundations
-- _ <- w ... "nbGroups" $ o.nbGroups
-- _ <- w ... "nbPeriods" $ o.nbPeriods
-- _ <- w ... "nbTerms" $ o.nbTerms
-- _ <- w ... "sources" $ o.sources
-- _ <- w ... "terms" $ {}
-- _ <- w ... "timeScale" $ o.timeScale
-- _ <- w ... "weighted" $ o.weighted
(freq :: Array Int) <- pure $ w .. "freq"
pure unit
-- forWithIndex_ o.foundations $ \i _ -> case maybeGetProperty (show i) freq of
-- Nothing -> freq ... (show i) $ 0
-- Just v -> freq ... (show i) $ (v + 1)
-- pure $ for o.groups \(g :: Group)-> pure unit
--------------------------------------------------------
--------------------------------------------------------
...
...
src/Gargantext/Components/PhyloExplorer/
layout
.js
→
src/Gargantext/Components/PhyloExplorer/
Types
.js
View file @
7a9fdad7
'use strict'
;
'use strict'
;
/**
function
readJson
(
file
,
callback
)
{
* @name yearToDate
var
raw
=
new
XMLHttpRequest
();
* @param {string} year
raw
.
overrideMimeType
(
"application/json"
);
* @returns {Date}
raw
.
open
(
"GET"
,
file
,
true
);
*/
raw
.
onreadystatechange
=
function
()
{
function
yearToDate
(
year
)
{
if
(
raw
.
readyState
===
4
&&
raw
.
status
==
"200"
)
{
var
d
=
new
Date
();
callback
(
raw
.
responseText
);
}
d
.
setYear
(
parseInt
(
year
));
}
d
.
setMonth
(
0
);
raw
.
send
(
null
);
d
.
setDate
(
1
);
}
return
d
;
function
unhide
(
mode
)
{
document
.
querySelector
(
"#reset"
).
style
.
visibility
=
"visible"
;
document
.
querySelector
(
"#label"
).
style
.
visibility
=
"visible"
;
document
.
querySelector
(
"#heading"
).
style
.
visibility
=
"visible"
;
if
(
mode
!=
"static"
)
{
document
.
querySelector
(
"#export"
).
style
.
visibility
=
"visible"
;
}
}
window
.
addEventListener
(
"load"
,
function
()
{
// read the config
readJson
(
"./config.json"
,
function
(
data1
){
var
conf
=
JSON
.
parse
(
data1
);
// available config modes are "static" or "explorable"
if
(
conf
.
mode
==
"static"
)
{
var
path
=
""
;
var
name
=
""
;
if
(
conf
.
path
==
null
||
conf
.
path
==
""
)
{
path
=
conf
.
defaultPath
;
name
=
conf
.
defaultName
;
}
else
{
path
=
conf
.
path
;
name
=
conf
.
pathName
;
}
document
.
querySelector
(
"#file-label"
).
style
.
display
=
"none"
;
document
.
querySelector
(
"#spin"
).
style
.
visibility
=
"visible"
;
document
.
getElementById
(
"phyloName"
).
innerHTML
=
name
;
document
.
querySelector
(
"#phyloName"
).
style
.
visibility
=
"visible"
;
readJson
(
path
,
function
(
data2
){
phylo
=
JSON
.
parse
(
data2
);
unhide
(
"static"
);
draw
(
phylo
);
})
}
})
})
function
readPhylo
(
file
)
{
var
reader
=
new
FileReader
();
reader
.
onload
=
(
function
(
f
)
{
return
function
(
e
)
{
try
{
json
=
JSON
.
parse
(
e
.
target
.
result
);
unhide
(
"explorable"
);
draw
(
json
)
}
catch
(
error
)
{
console
.
log
(
error
)
}
};
})(
file
);
reader
.
readAsText
(
file
,
"UTF-8"
);
}
}
/**
// display the Draw button after loading a phylo
* @name stringToDate
document
.
querySelector
(
"#file-path"
).
onchange
=
function
(){
* @param {string} str
document
.
querySelector
(
"#file-name"
).
textContent
=
(
this
.
files
[
0
].
name
).
substring
(
0
,
15
)
+
"..."
;
* @returns {Date}
// document.querySelector("#file-name").textContent = this.files[0].name;
*/
document
.
querySelector
(
"#draw"
).
style
.
display
=
"inline-block"
;
function
stringToDate
(
str
)
{
var
arr
=
(
str
.
replace
(
'"'
,
''
)).
split
(
'-'
);
var
d
=
new
Date
();
d
.
setYear
(
parseInt
(
arr
[
0
]));
d
.
setMonth
(
parseInt
(
arr
[
1
]));
d
.
setMonth
(
d
.
getMonth
()
-
1
);
d
.
setDate
(
parseInt
(
arr
[
2
]));
return
d
;
}
}
/**
// draw the phylo
* @name utcStringToDate
document
.
querySelector
(
"#draw"
).
onclick
=
function
()
{
* @param {string} str
document
.
querySelector
(
"#spin"
).
style
.
visibility
=
"visible"
;
* @returns {Date}
readPhylo
(
document
.
getElementById
(
"file-path"
).
files
[
0
]);
*/
function
utcStringToDate
(
str
)
{
var
arr
=
((
str
.
replace
(
'"'
,
''
)).
replace
(
' UTC'
,
''
)).
split
(
/
[\s
-:
]
+/
);
var
d
=
new
Date
();
d
.
setYear
(
parseInt
(
arr
[
0
]));
d
.
setMonth
(
parseInt
(
arr
[
1
]));
d
.
setDate
(
parseInt
(
arr
[
2
]));
d
.
setHours
(
parseInt
(
arr
[
3
]),
parseInt
(
arr
[
4
]),
parseInt
(
arr
[
5
]))
return
d
;
}
}
exports
.
yearToDate
=
yearToDate
;
function
drawPhyloInfo
(
elemClass
,
docs
,
foundations
,
branches
,
groups
,
terms
,
periods
)
{
exports
.
stringToDate
=
stringToDate
;
exports
.
utcStringToDate
=
utcStringToDate
;
ReactDOM
.
render
(
React
.
createElement
(
phyloCorpus
,{}),
document
.
getElementById
(
'phyloCorpus'
));
ReactDOM
.
render
(
React
.
createElement
(
phyloPhylo
,{}),
document
.
getElementById
(
'phyloPhylo'
));
ReactDOM
.
render
(
React
.
createElement
(
phyloHow
,{}),
document
.
getElementById
(
'phyloHow'
));
ReactDOM
.
render
(
React
.
createElement
(
phyloCorpusInfo
,{
nbDocs
:
docs
,
nbFoundations
:
foundations
,
nbPeriods
:
periods
}),
document
.
getElementById
(
'phyloCorpusInfo'
));
ReactDOM
.
render
(
React
.
createElement
(
phyloPhyloInfo
,{
nbTerms
:
terms
,
nbGroups
:
groups
,
nbBranches
:
branches
}),
document
.
getElementById
(
'phyloPhyloInfo'
));
}
function
draw
(
json
)
{
function
draw
(
json
)
{
// draw PhyloInfo
window
.
freq
=
{};
window
.
terms
=
{};
window
.
sources
=
[];
window
.
nbDocs
=
parseFloat
(
json
.
phyloDocs
);
window
.
nbBranches
=
parseFloat
(
json
.
phyloBranches
);
window
.
nbGroups
=
parseFloat
(
json
.
phyloGroups
);
window
.
nbTerms
=
parseFloat
(
json
.
phyloTerms
);
window
.
nbPeriods
=
parseFloat
(
json
.
phyloPeriods
);
window
.
nbFoundations
=
parseFloat
(
json
.
phyloFoundations
);
window
.
timeScale
=
json
.
phyloTimeScale
;
if
(
json
.
phyloSources
!=
undefined
)
{
var
sources
=
stringArrToArr
(
json
.
phyloSources
);
var
checkSource
=
document
.
getElementById
(
"checkSource"
);
for
(
var
i
=
0
;
i
<
sources
.
length
;
i
++
)
{
window
.
sources
.
push
({
source
:
sources
[
i
],
index
:
i
});
}
window
.
sources
.
sort
(
function
(
a
,
b
){
if
(
a
.
source
<
b
.
source
)
{
return
-
1
;
}
if
(
a
.
source
>
b
.
source
)
{
return
1
;
}
return
0
;
})
for
(
var
i
=
0
;
i
<
window
.
sources
.
length
;
i
++
)
{
var
option
=
document
.
createElement
(
"option"
);
option
.
text
=
window
.
sources
[
i
].
source
;
option
.
value
=
window
.
sources
[
i
].
index
;
checkSource
.
add
(
option
);
}
}
// original bounding box
bb
=
((
json
.
bb
).
split
(
','
)).
map
(
xy
=>
parseFloat
(
xy
))
drawPhyloInfo
(
""
,
window
.
nbDocs
,
window
.
nbFoundations
,
window
.
nbBranches
,
window
.
nbGroups
,
window
.
nbTerms
,
window
.
nbPeriods
)
// draw PhyloIsoline
var
branches
=
json
.
objects
.
filter
(
node
=>
node
.
nodeType
==
"branch"
).
map
(
function
(
b
){
return
{
x1
:
b
.
branch_x
,
y
:
b
.
branch_y
,
x2
:
parseFloat
(((
b
.
pos
).
split
(
','
))[
0
])
,
label
:
b
.
label
,
bId
:
parseInt
(
b
.
bId
),
gvid
:
parseInt
(
b
.
_gvid
)
}
});
var
periods
=
json
.
objects
.
filter
(
node
=>
node
.
nodeType
==
"period"
).
map
(
function
(
p
){
var
from
=
yearToDate
(
p
.
from
),
to
=
yearToDate
(
p
.
to
);
if
(
p
.
strFrom
!=
undefined
)
{
if
(
window
.
timeScale
==
"epoch"
)
{
from
=
utcStringToDate
(
p
.
strFrom
)
}
else
{
from
=
stringToDate
(
p
.
strFrom
)
}
}
if
(
p
.
strTo
!=
undefined
)
{
if
(
window
.
timeScale
==
"epoch"
)
{
to
=
utcStringToDate
(
p
.
strTo
)
}
else
{
to
=
stringToDate
(
p
.
strTo
)
}
}
return
{
from
:
from
,
to
:
to
,
y
:
parseFloat
(((
p
.
pos
).
split
(
','
))[
1
])}
});
// groups
// groups
window
.
weighted
=
false
;
window
.
weighted
=
false
;
...
...
src/Gargantext/Components/PhyloExplorer/Types.purs
View file @
7a9fdad7
module Gargantext.Components.PhyloExplorer.Types where
module Gargantext.Components.PhyloExplorer.Types
( PhyloDataSet(..)
, Branch, Period, Group
, parsePhyloJSONSet
) where
import Gargantext.Prelude
import Gargantext.Prelude
import Data.Array as Array
import Data.Date as Date
import Data.Generic.Rep (class Generic)
import Data.Generic.Rep (class Generic)
import Data.Generic.Rep as GR
import Data.Int as Int
import Data.Show.Generic (genericShow)
import Data.Maybe (Maybe(..), maybe)
import Gargantext.Utils.SimpleJSON (untaggedSumRep)
import Data.Number as Number
import Simple.JSON as JSON
import Data.String as String
import Data.Tuple as Tuple
import Data.Tuple.Nested ((/\))
type GraphData =
import Gargantext.Components.PhyloExplorer.JSON (PhyloJSONSet(..), PhyloObject(..))
( bb :: String
, color :: String
, fontsize :: String
-- @WIP Date or foreign?
, label :: String
foreign import yearToDate :: String -> Date.Date
, labelloc :: String
foreign import stringToDate :: String -> Date.Date
, lheight :: String
foreign import utcStringToDate :: String -> Date.Date
, lp :: String
, lwidth :: String
, name :: String
newtype PhyloDataSet = PhyloDataSet
, nodesep :: String
{ bb :: Array Number
, overlap :: String
, branches :: Array Branch
, phyloBranches :: String
, groups :: Array Group
, phyloDocs :: String
, nbBranches :: Int
, phyloFoundations :: String
, nbDocs :: Int
, phyloGroups :: String
, nbFoundations :: Int
, phyloPeriods :: String
, nbGroups :: Int
, phyloSources :: String
, nbPeriods :: Int
, phyloTerms :: String
, nbTerms :: Int
, phyloTimeScale :: String
, periods :: Array Period
, rank :: String
, sources :: Array String
, ranksep :: String
, timeScale :: String
, ratio :: String
, weighted :: Boolean
, splines :: String
, style :: String
)
--------------------------------------------------
newtype PhyloDataset = PhyloDataset
{ _subgraph_cnt :: Int
, directed :: Boolean
, edges :: Array Edge
, objects :: Array Object
, strict :: Boolean
| GraphData
}
}
derive instance Generic PhyloDataset _
derive instance Generic PhyloDataSet _
derive instance Eq PhyloDataset
instance Show PhyloDataset where show = genericShow
parsePhyloJSONSet :: PhyloJSONSet -> PhyloDataSet
derive newtype instance JSON.ReadForeign PhyloDataset
parsePhyloJSONSet (PhyloJSONSet o) = PhyloDataSet
{ bb : parseBB o.bb
--------------------------------------------------
, branches
, groups
type NodeData =
, nbBranches : parseInt o.phyloBranches
( height :: String
, nbDocs : parseInt o.phyloDocs
, label :: String
, nbFoundations : parseInt o.phyloFoundations
, name :: String
, nbGroups : parseInt o.phyloGroups
, nodeType :: String
, nbPeriods : parseInt o.phyloPeriods
, pos :: String
, nbTerms : parseInt o.phyloTerms
, shape :: String
, periods
, width :: String
, sources : parseSources o.phyloSources
)
, timeScale : o.phyloTimeScale
, weighted : getGlobalWeightedValue groups
data Object
}
= Layer
{ _gvid :: Int
where
, nodes :: Array Int
epochTS = o.phyloTimeScale == "epoch"
| GraphData
branches = parseBranches o.objects
}
groups = parseGroups epochTS o.objects
| BranchToNode
periods = parsePeriods epochTS o.objects
{ _gvid :: Int
, age :: String
-----------------------------------------------------------
, bId :: String
, birth :: String
data Branch = Branch
, branchId :: String
{ bId :: Int
, branch_x :: String
, gvid :: Int
, branch_y :: String
, label :: String
, fillcolor :: String
, x1 :: String
, fontname :: String
, x2 :: Number
, fontsize :: String
, y :: String
, size :: String
}
, style :: String
| NodeData
parseBranches :: Array PhyloObject -> Array Branch
}
parseBranches
| GroupToNode
= map parse
{ _gvid :: Int
>>> Array.catMaybes
, bId :: String
, branchId :: String
where
, fontname :: String
parse :: PhyloObject -> Maybe Branch
, foundation :: String
parse (BranchToNode o) = Just $ Branch
, frequence :: String
{ bId : parseInt o.bId
, from :: String
, gvid : o._gvid
, lbl :: String
, label : o.label
, penwidth :: String
, x1 : o.branch_x
, role :: String
, x2 : Tuple.fst $ parsePos o.pos
, seaLvl :: String
, y : o.branch_y
, source :: String
}
, strFrom :: String
parse _ = Nothing
, strTo :: String
, support :: String
-----------------------------------------------------------
, to :: String
, weight :: String
data Period = Period
| NodeData
{ from :: Date.Date
}
, to :: Date.Date
| PeriodToNode
, y :: Number
{ _gvid :: Int
}
, fontsize :: String
, from :: String
parsePeriods :: Boolean -> Array PhyloObject -> Array Period
, strFrom :: String
parsePeriods epoch
, strTo :: String
= map parse
, to :: String
>>> Array.catMaybes
| NodeData
}
where
parse :: PhyloObject -> Maybe Period
derive instance Generic Object _
parse (PeriodToNode o) = Just $ Period
derive instance Eq Object
{ from : parseNodeDate o.strFrom o.from epoch
instance Show Object where show = genericShow
, to : parseNodeDate o.strTo o.to epoch
instance JSON.ReadForeign Object where
, y : Tuple.snd $ parsePos o.pos
readImpl f = GR.to <$> untaggedSumRep f
}
parse _ = Nothing
--------------------------------------------------
-----------------------------------------------------------
type EdgeData =
data Group = Group
( color :: String
{ bId :: Int
, head :: Int
, foundation :: Array Int -- @WIP: Array String ???
, pos :: String
, from :: Date.Date
, tail :: Int
, gId :: Int
, width :: String
, label :: Array String
)
, role :: Array Int
, size :: Int
data Edge
, source :: Array String
= GroupToGroup
, to :: Date.Date
{ _gvid :: Int
, weight :: Number
, constraint :: String
, x :: Number
, edgeType :: String
, y :: Number
, lbl :: String
}
, penwidth :: String
| EdgeData
parseGroups :: Boolean -> Array PhyloObject -> Array Group
}
parseGroups epoch
| BranchToGroup
= map parse
{ _gvid :: Int
>>> Array.catMaybes
, arrowhead :: String
, edgeType :: String
where
| EdgeData
parse :: PhyloObject -> Maybe Group
}
parse (GroupToNode o) = Just $ Group
| BranchToBranch
{ from : parseNodeDate o.strFrom o.from epoch
{ _gvid :: Int
, to : parseNodeDate o.strTo o.to epoch
, arrowhead :: String
, x : Tuple.fst $ parsePos o.pos
, style :: String
, y : Tuple.snd $ parsePos o.pos
| EdgeData
, bId : parseInt o.bId
}
, gId : o._gvid
| GroupToAncestor
, size : parseInt o.support
{ _gvid :: Int
, source: parseSources o.source
, arrowhead :: String
, weight: stringedMaybeToNumber o.weight
, lbl :: String
, label : stringedArrayToArray o.lbl
, penwidth :: String
, role : stringedArrayToArray' o.role
, style :: String
, foundation: stringedArrayToArray' o.foundation
| EdgeData
}
}
parse _ = Nothing
| PeriodToPeriod
{ _gvid :: Int
-----------------------------------------------------------
| EdgeData
}
parseInt :: String -> Int
parseInt s = maybe 0 identity $ Int.fromString s
derive instance Generic Edge _
derive instance Eq Edge
parseFloat :: String -> Number
instance Show Edge where show = genericShow
parseFloat s = maybe 0.0 identity $ Number.fromString s
instance JSON.ReadForeign Edge where
readImpl f = GR.to <$> untaggedSumRep f
parseSources :: String -> Array String
parseSources
= String.replace (String.Pattern "[") (String.Replacement "")
>>> String.replace (String.Pattern "]") (String.Replacement "")
>>> String.split (String.Pattern ",")
>>> Array.filter (\s -> not eq 0 $ String.length s)
>>> Array.sort
parseBB :: String -> Array Number
parseBB
= String.split (String.Pattern ",")
>>> map parseFloat
parseNodeDate :: Maybe String -> String -> Boolean -> Date.Date
parseNodeDate Nothing year _ = yearToDate(year)
parseNodeDate (Just str) _ true = utcStringToDate(str)
parseNodeDate (Just str) _ false = stringToDate(str)
parsePos :: String -> Tuple.Tuple Number Number
parsePos
= String.split (String.Pattern ",")
>>> \a -> (p $ Array.index a 0) /\
(p $ Array.index a 1)
where
p = case _ of
Nothing -> 0.0
Just s -> parseFloat s
-- @WIP: why taking last value? use `any`?
getGlobalWeightedValue :: Array Group -> Boolean
getGlobalWeightedValue
= Array.last
>>> case _ of
Nothing -> false
Just (Group o) -> o.weight > 0.0
stringedMaybeToNumber :: String -> Number
stringedMaybeToNumber "Nothing" = 0.0
stringedMaybeToNumber s =
s # String.replace (String.Pattern "Just ") (String.Replacement "")
>>> parseFloat
stringedArrayToArray :: String -> Array String
stringedArrayToArray str
= str # String.length
>>> (\length -> String.splitAt (length - 1) str)
>>> (\{ after } -> String.splitAt 1 after)
>>> (\{ after } -> String.split (String.Pattern "|") after)
>>> map String.trim
stringedArrayToArray' :: String -> Array Int
stringedArrayToArray'
= stringedArrayToArray
>>> map parseInt
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment