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
133
Issues
133
List
Board
Labels
Milestones
Merge Requests
5
Merge Requests
5
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
06a07736
Unverified
Commit
06a07736
authored
Nov 19, 2018
by
Nicolas Pouillard
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[DOCS] Some fixes for favorites and deletes
parent
ce5135f4
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
63 additions
and
119 deletions
+63
-119
REST.purs
src/Gargantext/Config/REST.purs
+5
-0
Documents.purs
src/Gargantext/Pages/Corpus/Tabs/Documents.purs
+56
-115
States.purs
src/Gargantext/Pages/Corpus/Tabs/States.purs
+2
-4
No files found.
src/Gargantext/Config/REST.purs
View file @
06a07736
...
@@ -51,5 +51,10 @@ put url = send PUT url <<< Just
...
@@ -51,5 +51,10 @@ put url = send PUT url <<< Just
delete :: forall a. DecodeJson a => String -> Aff a
delete :: forall a. DecodeJson a => String -> Aff a
delete url = send DELETE url noReqBody
delete url = send DELETE url noReqBody
-- This might not be a good idea:
-- https://stackoverflow.com/questions/14323716/restful-alternatives-to-delete-request-body
deleteWithBody :: forall a b. EncodeJson a => DecodeJson b => String -> a -> Aff b
deleteWithBody url = send DELETE url <<< Just
post :: forall a b. EncodeJson a => DecodeJson b => String -> a -> Aff b
post :: forall a b. EncodeJson a => DecodeJson b => String -> a -> Aff b
post url = send POST url <<< Just
post url = send POST url <<< Just
src/Gargantext/Pages/Corpus/Tabs/Documents.purs
View file @
06a07736
...
@@ -9,38 +9,36 @@ import Affjax.ResponseFormat (printResponseFormatError)
...
@@ -9,38 +9,36 @@ import Affjax.ResponseFormat (printResponseFormatError)
import Affjax.ResponseFormat as ResponseFormat
import Affjax.ResponseFormat as ResponseFormat
import Control.Monad.Cont.Trans (lift)
import Control.Monad.Cont.Trans (lift)
import Data.Argonaut (class DecodeJson, class EncodeJson, decodeJson, encodeJson, jsonEmptyObject, (.?), (:=), (~>))
import Data.Argonaut (class DecodeJson, class EncodeJson, decodeJson, encodeJson, jsonEmptyObject, (.?), (:=), (~>))
import Data.Array (
cons,
drop, take, (:))
import Data.Array (drop, take, (:))
import Data.Either (Either(..))
import Data.Either (Either(..))
import Data.Generic.Rep (class Generic)
import Data.Generic.Rep (class Generic)
import Data.Generic.Rep.Show (genericShow)
import Data.Generic.Rep.Show (genericShow)
import Data.HTTP.Method (Method(..))
import Data.HTTP.Method (Method(..))
import Data.Maybe (Maybe(..), maybe)
import Data.Maybe (Maybe(..), maybe)
import Data.Set (Set)
import Data.Set as Set
import Data.Tuple (Tuple(..))
import Data.Tuple (Tuple(..))
import Effect (Effect)
import Effect.Aff (Aff)
import Effect.Aff (Aff)
import Effect.Class (liftEffect)
import Effect.Class (liftEffect)
import React as React
import React as React
import React (ReactClass, ReactElement, Children)
import React (ReactClass, ReactElement, Children)
import React.DOM (a, br', div, input, p, text)
import React.DOM.Props (_type, className, href, style, placeholder, name)
import Thermite (Render, Spec, defaultPerformAction, simpleSpec)
------------------------------------------------------------------------
------------------------------------------------------------------------
import Gargantext.Prelude
import Gargantext.Prelude
import Gargantext.Config (NodeType(..), TabType(..), toUrl, End(..), OrderBy(..))
import Gargantext.Config (NodeType(..), TabType(..), toUrl, End(..), OrderBy(..))
import Gargantext.Config.REST (get, p
ost
)
import Gargantext.Config.REST (get, p
ut, post, deleteWithBody
)
import Gargantext.Utils.DecodeMaybe ((.|))
import Gargantext.Utils.DecodeMaybe ((.|))
import Gargantext.Components.Charts.Options.ECharts (chart)
import Gargantext.Components.Charts.Options.ECharts (chart)
import Gargantext.Components.Loader as Loader
import Gargantext.Components.Loader as Loader
import Gargantext.Components.Table as T
import Gargantext.Components.Table as T
import Gargantext.Components.Node (NodePoly(..))
import Gargantext.Components.Node (NodePoly(..))
import Gargantext.Components.Table as T
import Gargantext.Components.Table as T
import Gargantext.Config (NodeType(..), TabType(..), toUrl, End(..), OrderBy(..))
import Gargantext.Config.REST (get, post)
import Gargantext.Pages.Corpus.Dashboard (globalPublis)
import Gargantext.Pages.Corpus.Dashboard (globalPublis)
import Gargantext.Pages.Corpus.Tabs.Types (CorpusInfo(..), Props)
import Gargantext.Pages.Corpus.Tabs.Types (CorpusInfo(..), Props)
import Gargantext.Utils.DecodeMaybe ((.|))
import Gargantext.Utils.DecodeMaybe ((.|))
import React.DOM (a, br', button, div, i, input, p, text)
import React.DOM (a, br', button, div, i, input, p, text)
import React.DOM.Props (_type, className, href, name, onClick, placeholder, style, value)
import React.DOM.Props (_type, className, href, name, onClick, placeholder, style, value)
import Thermite (PerformAction, Render, Spec, defaultPerformAction, modifyState, simpleSpec)
import Thermite (PerformAction, Render, Spec, defaultPerformAction, modifyState
_
, simpleSpec)
------------------------------------------------------------------------
------------------------------------------------------------------------
-- TODO: Pagination Details are not available from the BackEnd
-- TODO: Pagination Details are not available from the BackEnd
-- TODO: Search is pending
-- TODO: Search is pending
...
@@ -49,51 +47,15 @@ import Thermite (PerformAction, Render, Spec, defaultPerformAction, modifyState,
...
@@ -49,51 +47,15 @@ import Thermite (PerformAction, Render, Spec, defaultPerformAction, modifyState,
-- TODO: Filter is Pending
-- TODO: Filter is Pending
-- TODO: When a pagination link is clicked, reload data.
-- TODO: When a pagination link is clicked, reload data.
type State =
type State =
{ documents :: DocumentsView
{ documentIdsToDelete :: Set Int
, deleteRows :: Boolean
, deleteRowId :: Array Int
, delete :: Array DeleteRow
}
newtype DeleteRow = DeleteRow
{
deleteRowId :: Int
, deleteRows :: Boolean
}
}
data Action
data Action
=
SendFavorites Int
=
MarkFavorites (Array Int)
|
DeleteDocuments
Int
|
ToggleDocumentToDelete
Int
| Trash
| Trash
performAction :: PerformAction State Props Action
performAction (SendFavorites nid) {path : nodeId} _ = void $ do
s' <- lift $ favorites nid (FavoriteQuery {favorites : [nid]})
case s' of
Left err -> do
logs err
modifyState identity
Right d -> modifyState identity
--TODO add array of delete rows here
performAction (DeleteDocuments nid) _ s = void $ do
logs $ show nid
modifyState \state -> state {deleteRowId = ( cons nid s.deleteRowId), deleteRows = true}
performAction Trash {path:nodeId} state = void $ do
s' <- lift $ deleteDocuments nodeId (DeleteDocumentQuery {documents : state.deleteRowId})
case s' of
Left err -> do
logs err
modifyState identity
Right d -> modifyState identity
-- modifyState \state -> state {deleteRowId = ( cons nid s.deleteRowId), deleteRows = true} for referrence
newtype DocumentsView
newtype DocumentsView
= DocumentsView
= DocumentsView
{ _id :: Int
{ _id :: Int
...
@@ -176,6 +138,22 @@ filterSpec = simpleSpec defaultPerformAction render
...
@@ -176,6 +138,22 @@ filterSpec = simpleSpec defaultPerformAction render
layoutDocview :: Spec State Props Action
layoutDocview :: Spec State Props Action
layoutDocview = simpleSpec performAction render
layoutDocview = simpleSpec performAction render
where
where
performAction :: PerformAction State Props Action
performAction (MarkFavorites nids) {path : nodeId} _ =
void $ lift $ putFavorites nodeId (FavoriteQuery {favorites: nids})
--TODO add array of delete rows here
performAction (ToggleDocumentToDelete nid) _ _ =
modifyState_ \state -> state {documentIdsToDelete = toggleSet nid state.documentIdsToDelete}
performAction Trash {path: nodeId} {documentIdsToDelete} =
void $ lift $ deleteDocuments nodeId (DeleteDocumentQuery {documents: Set.toUnfoldable documentIdsToDelete})
-- TODO: what to do now that the documents are deleted
-- * should we reload?
-- * should we locally update our data?
-- * should we reset documentIdsToDelete?
-- * if so, how to un-check the checkboxes since the inputs are uncontrolled?
-- (maybe there is no need to uncheck them if they disapear because we
-- either reload or local update our data)
render :: Render State Props Action
render :: Render State Props Action
render dispatch {path: nodeId, loaded: corpusInfo} _ _ =
render dispatch {path: nodeId, loaded: corpusInfo} _ _ =
[ p [] []
[ p [] []
...
@@ -188,17 +166,16 @@ layoutDocview = simpleSpec performAction render
...
@@ -188,17 +166,16 @@ layoutDocview = simpleSpec performAction render
[ pageLoader
[ pageLoader
{ path: initialPageParams nodeId
{ path: initialPageParams nodeId
, corpusInfo
, corpusInfo
, dispatch
}
}
]
]
, div [className "col-md-12"]
, div [className "col-md-12"]
[ button [style {backgroundColor: "peru", padding : "9px", color : "white", border : "white", float: "right"}
[ button [
style {backgroundColor: "peru", padding : "9px", color : "white", border : "white", float: "right"}
, onClick $ (\_ -> dispatch
$ Trash
)
, onClick $ (\_ -> dispatch
Trash
)
]
]
[ i [className "glyphitem glyphicon glyphicon-trash", style {marginRight : "9px"}] []
[ i [className "glyphitem glyphicon glyphicon-trash", style {marginRight : "9px"}] []
, text "Trash it !"
, text "Trash it !"
]
]
]
]
]
]
]
]
...
@@ -242,20 +219,25 @@ loadPage {nodeId, params: {limit, offset, orderBy}} = do
...
@@ -242,20 +219,25 @@ loadPage {nodeId, params: {limit, offset, orderBy}} = do
convOrderBy _ = DateAsc -- TODO
convOrderBy _ = DateAsc -- TODO
type PageLoaderProps =
type PageLoaderProps
ext
=
{ path :: PageParams
{ path :: PageParams
, corpusInfo :: Maybe (NodePoly CorpusInfo)
, corpusInfo :: Maybe (NodePoly CorpusInfo)
, dispatch :: Action -> Effect Unit
| ext
}
}
renderPage :: forall props path.
renderPage :: forall props path.
Render (Loader.State {nodeId :: Int | path} (Array DocumentsView))
Render (Loader.State {nodeId :: Int | path} (Array DocumentsView))
{corpusInfo :: Maybe (NodePoly CorpusInfo) | props}
{ corpusInfo :: Maybe (NodePoly CorpusInfo)
, dispatch :: Action -> Effect Unit
| props
}
(Loader.Action PageParams)
(Loader.Action PageParams)
renderPage _ _ {loaded: Nothing} _ = [] -- TODO loading spinner
renderPage _ _ {loaded: Nothing} _ = [] -- TODO loading spinner
renderPage
dispatch {corpusInfo
} {currentPath: {nodeId}, loaded: Just res} _ =
renderPage
loaderDispatch {corpusInfo, dispatch
} {currentPath: {nodeId}, loaded: Just res} _ =
[ T.tableElt
[ T.tableElt
{ rows
{ rows
, setParams: \params -> liftEffect $
d
ispatch (Loader.SetPath {nodeId, params})
, setParams: \params -> liftEffect $
loaderD
ispatch (Loader.SetPath {nodeId, params})
, container: T.defaultContainer { title: "Documents" }
, container: T.defaultContainer { title: "Documents" }
, colNames:
, colNames:
T.ColumnName <$>
T.ColumnName <$>
...
@@ -274,13 +256,13 @@ renderPage dispatch {corpusInfo} {currentPath: {nodeId}, loaded: Just res} _ =
...
@@ -274,13 +256,13 @@ renderPage dispatch {corpusInfo} {currentPath: {nodeId}, loaded: Just res} _ =
}
}
]
]
where
where
dispatch2 _ = logs "TODO dispatch2"
fa true = "fas "
fa true = "fas "
fa false = "far "
fa false = "far "
rows = (\(DocumentsView r) ->
rows = (\(DocumentsView r) ->
{ row:
{ row:
[ div []
[ div []
[ a [className $ fa r.fav <> "fa-star" ,onClick $ (\_-> dispatch2 $ (SendFavorites r._id))] []
[ a [className $ fa r.fav <> "fa-star" ,onClick $ (\_->
dispatch $ MarkFavorites [r._id])] []
]
]
-- TODO show date: Year-Month-Day only
-- TODO show date: Year-Month-Day only
, if (r.delete) then
, if (r.delete) then
...
@@ -295,15 +277,15 @@ renderPage dispatch {corpusInfo} {currentPath: {nodeId}, loaded: Just res} _ =
...
@@ -295,15 +277,15 @@ renderPage dispatch {corpusInfo} {currentPath: {nodeId}, loaded: Just res} _ =
div [style {textDecoration : "line-through"}] [ text r.source]
div [style {textDecoration : "line-through"}] [ text r.source]
else
else
div [] [ text r.source]
div [] [ text r.source]
, input [ _type "checkbox", onClick $ (\_ -> dispatch
2 $ (DeleteDocuments r._id)
)]
, input [ _type "checkbox", onClick $ (\_ -> dispatch
$ ToggleDocumentToDelete r._id
)]
]
]
, delete: true
, delete: true
}) <$> res
}) <$> res
pageLoaderClass :: ReactClass
{ path :: PageParams, corpusInfo :: Maybe (NodePoly CorpusInfo), children :: Children }
pageLoaderClass :: ReactClass
(PageLoaderProps (children :: Children))
pageLoaderClass = Loader.createLoaderClass' "PageLoader" loadPage renderPage
pageLoaderClass = Loader.createLoaderClass' "PageLoader" loadPage renderPage
pageLoader :: PageLoaderProps -> ReactElement
pageLoader :: PageLoaderProps
()
-> ReactElement
pageLoader props = React.createElement pageLoaderClass props []
pageLoader props = React.createElement pageLoaderClass props []
---------------------------------------------------------
---------------------------------------------------------
...
@@ -358,58 +340,17 @@ instance encodeJsonDDQuery :: EncodeJson DeleteDocumentQuery where
...
@@ -358,58 +340,17 @@ instance encodeJsonDDQuery :: EncodeJson DeleteDocumentQuery where
= "documents" := post.documents
= "documents" := post.documents
~> jsonEmptyObject
~> jsonEmptyObject
putFavorites :: Int -> FavoriteQuery -> Aff (Array Int)
putFavorites nodeId = put (toUrl Back Node nodeId <> "/favorites")
deleteFavorites :: Int -> FavoriteQuery -> Aff (Array Int)
deleteFavorites nodeId = deleteWithBody (toUrl Back Node nodeId <> "/favorites")
deleteDocuments :: Int -> DeleteDocumentQuery -> Aff (Array Int)
deleteDocuments nodeId = deleteWithBody (toUrl Back Node nodeId <> "/documents")
favorites :: Int -> FavoriteQuery -> Aff (Either String (Array Int))
-- TODO: not optimal but Data.Set lacks some function (Set.alter)
favorites nodeId reqbody= do
toggleSet :: forall a. Ord a => a -> Set a -> Set a
res <- request $ defaultRequest
toggleSet a s
{ url = "http://localhost:8008/api/v1.0/node/"<>show nodeId <>"/favorites"
| Set.member a s = Set.delete a s
, responseFormat = ResponseFormat.json
| otherwise = Set.insert a s
, method = Left PUT
, headers = []
, content = Just $ Json $ encodeJson reqbody
}
case res.body of
Left err -> do
logs $ printResponseFormatError err
pure $ Left $ printResponseFormatError err
Right json -> do
let obj = decodeJson json
pure obj
deleteFavorites :: Int -> FavoriteQuery -> Aff (Either String (Array Int))
deleteFavorites nodeId reqbody = do
-- TODO use Config.REST.delete
res <- request $ defaultRequest
{ url = "http://localhost:8008/api/v1.0/node/"<>show nodeId<>"/favorites"
, responseFormat = ResponseFormat.json
, method = Left DELETE
, headers = []
, content = Just $ Json $ encodeJson reqbody
}
case res.body of
Left err -> do
logs $ printResponseFormatError err
pure $ Left $ printResponseFormatError err
Right json -> do
let obj = decodeJson json
pure obj
deleteDocuments :: Int -> DeleteDocumentQuery -> Aff (Either String Unit)
deleteDocuments nodeId reqbody= do
-- TODO use Config.REST.delete
res <- request $ defaultRequest
{ url = "http://localhost:8008/api/v1.0/annuaire/"<>show nodeId <>"/documents"
, responseFormat = ResponseFormat.json
, method = Left DELETE
, headers = []
, content = Just $ Json $ encodeJson reqbody
}
case res.body of
Left err -> do
logs $ printResponseFormatError err
pure $ Left $ printResponseFormatError err
Right json -> do
-- let obj = decodeJson json
pure $ Right unit
src/Gargantext/Pages/Corpus/Tabs/States.purs
View file @
06a07736
module Gargantext.Pages.Corpus.Tabs.States where
module Gargantext.Pages.Corpus.Tabs.States where
import Data.Lens (Lens', lens)
import Data.Lens (Lens', lens)
import Gargantext.Prelude
import Gargantext.Pages.Corpus.Tabs.Documents as D
import Gargantext.Pages.Corpus.Tabs.Documents as D
import Gargantext.Pages.Corpus.Tabs.Ngrams.NgramsTable as N
import Gargantext.Pages.Corpus.Tabs.Ngrams.NgramsTable as N
import Gargantext.Components.Tab as Tab
import Gargantext.Components.Tab as Tab
...
@@ -15,10 +16,7 @@ type State =
...
@@ -15,10 +16,7 @@ type State =
initialState :: {} -> State
initialState :: {} -> State
initialState _ =
initialState _ =
{ docsView :
{ docsView :
{ documents : D.sampleData'
{ documentIdsToDelete : mempty
, deleteRows : false
, deleteRowId : []
, delete : []
}
}
, ngramsView : {} -- N.initialState
, ngramsView : {} -- N.initialState
, activeTab : 0
, activeTab : 0
...
...
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