Merge pull request #492 from eikek/person-org

Person org
This commit is contained in:
mergify[bot] 2020-12-02 00:36:38 +00:00 committed by GitHub
commit 7e4c0b3e42
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
24 changed files with 523 additions and 146 deletions

View File

@ -28,8 +28,10 @@ trait OOrganization[F[_]] {
def findAllPersonRefs(account: AccountId, nameQuery: Option[String]): F[Vector[IdRef]] def findAllPersonRefs(account: AccountId, nameQuery: Option[String]): F[Vector[IdRef]]
/** Add a new person with their contacts. The additional organization is ignored. */
def addPerson(s: PersonAndContacts): F[AddResult] def addPerson(s: PersonAndContacts): F[AddResult]
/** Update a person with their contacts. The additional organization is ignored. */
def updatePerson(s: PersonAndContacts): F[AddResult] def updatePerson(s: PersonAndContacts): F[AddResult]
def deleteOrg(orgId: Ident, collective: Ident): F[AddResult] def deleteOrg(orgId: Ident, collective: Ident): F[AddResult]
@ -41,7 +43,11 @@ object OOrganization {
case class OrgAndContacts(org: ROrganization, contacts: Seq[RContact]) case class OrgAndContacts(org: ROrganization, contacts: Seq[RContact])
case class PersonAndContacts(person: RPerson, contacts: Seq[RContact]) case class PersonAndContacts(
person: RPerson,
org: Option[ROrganization],
contacts: Seq[RContact]
)
def apply[F[_]: Effect](store: Store[F]): Resource[F, OOrganization[F]] = def apply[F[_]: Effect](store: Store[F]): Resource[F, OOrganization[F]] =
Resource.pure[F, OOrganization[F]](new OOrganization[F] { Resource.pure[F, OOrganization[F]](new OOrganization[F] {
@ -79,14 +85,14 @@ object OOrganization {
): F[Vector[PersonAndContacts]] = ): F[Vector[PersonAndContacts]] =
store store
.transact(QOrganization.findPersonAndContact(account.collective, query, _.name)) .transact(QOrganization.findPersonAndContact(account.collective, query, _.name))
.map({ case (person, cont) => PersonAndContacts(person, cont) }) .map({ case (person, org, cont) => PersonAndContacts(person, org, cont) })
.compile .compile
.toVector .toVector
def findPerson(account: AccountId, persId: Ident): F[Option[PersonAndContacts]] = def findPerson(account: AccountId, persId: Ident): F[Option[PersonAndContacts]] =
store store
.transact(QOrganization.getPersonAndContact(account.collective, persId)) .transact(QOrganization.getPersonAndContact(account.collective, persId))
.map(_.map({ case (org, cont) => PersonAndContacts(org, cont) })) .map(_.map({ case (pers, org, cont) => PersonAndContacts(pers, org, cont) }))
def findAllPersonRefs( def findAllPersonRefs(
account: AccountId, account: AccountId,

View File

@ -0,0 +1,18 @@
package docspell.common
import io.circe._
import io.circe.generic.semiauto._
case class PersonRef(id: Ident, name: String, organization: Option[Ident]) {
def toIdRef: IdRef =
IdRef(id, name)
}
object PersonRef {
implicit val jsonEncoder: Encoder[PersonRef] =
deriveEncoder[PersonRef]
implicit val jsonDecoder: Decoder[PersonRef] =
deriveDecoder[PersonRef]
}

View File

@ -6,8 +6,8 @@ import cats.effect.Sync
import cats.implicits._ import cats.implicits._
import docspell.common._ import docspell.common._
import docspell.joex.scheduler.Task import docspell.joex.scheduler.{Context, Task}
import docspell.store.records.RAttachmentMeta import docspell.store.records.{RAttachmentMeta, RPerson}
/** Calculate weights for candidates that adds the most likely /** Calculate weights for candidates that adds the most likely
* candidate a lower number. * candidate a lower number.
@ -15,21 +15,40 @@ import docspell.store.records.RAttachmentMeta
object EvalProposals { object EvalProposals {
def apply[F[_]: Sync](data: ItemData): Task[F, ProcessItemArgs, ItemData] = def apply[F[_]: Sync](data: ItemData): Task[F, ProcessItemArgs, ItemData] =
Task { _ => Task { ctx =>
Timestamp for {
.current[F] now <- Timestamp.current[F]
.map { now => personRefs <- findOrganizationRelation[F](data, ctx)
val metas = data.metas.map(calcCandidateWeight(now.toUtcDate)) metas = data.metas.map(calcCandidateWeight(now.toUtcDate, personRefs))
data.copy(metas = metas) } yield data.copy(metas = metas)
}
} }
def calcCandidateWeight(now: LocalDate)(rm: RAttachmentMeta): RAttachmentMeta = { def findOrganizationRelation[F[_]: Sync](
val list = rm.proposals.change(mp => mp.addWeights(weight(rm, mp, now))) data: ItemData,
ctx: Context[F, _]
): F[Map[Ident, PersonRef]] = {
val corrPersIds = data.metas
.flatMap(_.proposals.find(MetaProposalType.CorrPerson))
.flatMap(_.values.toList.map(_.ref.id))
.toSet
ctx.store
.transact(RPerson.findOrganization(corrPersIds))
.map(_.map(p => (p.id, p)).toMap)
}
def calcCandidateWeight(now: LocalDate, personRefs: Map[Ident, PersonRef])(
rm: RAttachmentMeta
): RAttachmentMeta = {
val list = rm.proposals.change(mp => mp.addWeights(weight(rm, mp, now, personRefs)))
rm.copy(proposals = list.sortByWeights) rm.copy(proposals = list.sortByWeights)
} }
def weight(rm: RAttachmentMeta, mp: MetaProposal, ref: LocalDate)( def weight(
rm: RAttachmentMeta,
mp: MetaProposal,
ref: LocalDate,
personRefs: Map[Ident, PersonRef]
)(
cand: MetaProposal.Candidate cand: MetaProposal.Candidate
): Double = ): Double =
mp.proposalType match { mp.proposalType match {
@ -51,7 +70,27 @@ object EvalProposals {
val words = cand.origin.map(_.label.split(' ').length).max.toDouble val words = cand.origin.map(_.label.split(' ').length).max.toDouble
val nerFac = val nerFac =
cand.origin.map(label => nerTagFactor(label.tag, mp.proposalType)).min cand.origin.map(label => nerTagFactor(label.tag, mp.proposalType)).min
(1 / words) * (1 / tagCount) * positionWeight(pos, textLen) * nerFac val corrPerFac = corrOrgPersonFactor(rm, mp, personRefs, cand)
(1 / words) * (1 / tagCount) * positionWeight(pos, textLen) * nerFac * corrPerFac
}
def corrOrgPersonFactor(
rm: RAttachmentMeta,
mp: MetaProposal,
personRefs: Map[Ident, PersonRef],
cand: MetaProposal.Candidate
): Double =
mp.proposalType match {
case MetaProposalType.CorrPerson =>
(for {
currentOrg <- rm.proposals
.find(MetaProposalType.CorrOrg)
.map(_.values.head.ref.id)
personOrg <- personRefs.get(cand.ref.id).flatMap(_.organization)
fac = if (currentOrg == personOrg) 0.5 else 1
} yield fac).getOrElse(1)
case _ =>
1
} }
def positionWeight(pos: Int, total: Int): Double = def positionWeight(pos: Int, total: Int): Double =

View File

@ -4833,6 +4833,8 @@ components:
format: ident format: ident
name: name:
type: string type: string
organization:
$ref: "#/components/schemas/IdName"
address: address:
$ref: "#/components/schemas/Address" $ref: "#/components/schemas/Address"
contacts: contacts:

View File

@ -414,15 +414,16 @@ trait Conversions {
} }
def mkPerson(v: OOrganization.PersonAndContacts): Person = { def mkPerson(v: OOrganization.PersonAndContacts): Person = {
val ro = v.person val rp = v.person
Person( Person(
ro.pid, rp.pid,
ro.name, rp.name,
Address(ro.street, ro.zip, ro.city, ro.country), v.org.map(o => IdName(o.oid, o.name)),
Address(rp.street, rp.zip, rp.city, rp.country),
v.contacts.map(mkContact).toList, v.contacts.map(mkContact).toList,
ro.notes, rp.notes,
ro.concerning, rp.concerning,
ro.created rp.created
) )
} }
@ -433,7 +434,7 @@ trait Conversions {
now <- Timestamp.current[F] now <- Timestamp.current[F]
pid <- Ident.randomId[F] pid <- Ident.randomId[F]
cont <- contacts(pid) cont <- contacts(pid)
org = RPerson( pers = RPerson(
pid, pid,
cid, cid,
v.name, v.name,
@ -444,9 +445,10 @@ trait Conversions {
v.notes, v.notes,
v.concerning, v.concerning,
now, now,
now now,
v.organization.map(_.id)
) )
} yield OOrganization.PersonAndContacts(org, cont) } yield OOrganization.PersonAndContacts(pers, None, cont)
} }
def changePerson[F[_]: Sync]( def changePerson[F[_]: Sync](
@ -458,7 +460,7 @@ trait Conversions {
for { for {
now <- Timestamp.current[F] now <- Timestamp.current[F]
cont <- contacts(v.id) cont <- contacts(v.id)
org = RPerson( pers = RPerson(
v.id, v.id,
cid, cid,
v.name, v.name,
@ -469,9 +471,10 @@ trait Conversions {
v.notes, v.notes,
v.concerning, v.concerning,
v.created, v.created,
now now,
v.organization.map(_.id)
) )
} yield OOrganization.PersonAndContacts(org, cont) } yield OOrganization.PersonAndContacts(pers, None, cont)
} }
// contact // contact

View File

@ -0,0 +1,7 @@
ALTER TABLE "person"
ADD COLUMN "oid" varchar(254);
ALTER TABLE "person"
ADD CONSTRAINT fk_person_organization
FOREIGN KEY ("oid")
REFERENCES "organization"("oid");

View File

@ -0,0 +1,7 @@
ALTER TABLE `person`
ADD COLUMN `oid` varchar(254);
ALTER TABLE `person`
ADD CONSTRAINT fk_person_organization
FOREIGN KEY (`oid`)
REFERENCES `organization`(`oid`);

View File

@ -0,0 +1,7 @@
ALTER TABLE "person"
ADD COLUMN "oid" varchar(254);
ALTER TABLE "person"
ADD CONSTRAINT fk_person_organization
FOREIGN KEY ("oid")
REFERENCES "organization"("oid");

View File

@ -41,7 +41,7 @@ object QOrganization {
Seq.empty Seq.empty
}) })
(selectSimple(cols, from, and(q)) ++ orderBy(order(OC).f)) (selectSimple(cols, from, and(q)) ++ orderBy(order(OC).prefix("o").f))
.query[(ROrganization, Option[RContact])] .query[(ROrganization, Option[RContact])]
.stream .stream
.groupAdjacentBy(_._1) .groupAdjacentBy(_._1)
@ -82,17 +82,21 @@ object QOrganization {
coll: Ident, coll: Ident,
query: Option[String], query: Option[String],
order: PC.type => Column order: PC.type => Column
): Stream[ConnectionIO, (RPerson, Vector[RContact])] = { ): Stream[ConnectionIO, (RPerson, Option[ROrganization], Vector[RContact])] = {
val pColl = PC.cid.prefix("p") val pColl = PC.cid.prefix("p")
val pName = RPerson.Columns.name.prefix("p") val pName = RPerson.Columns.name.prefix("p")
val pNotes = RPerson.Columns.notes.prefix("p") val pNotes = RPerson.Columns.notes.prefix("p")
val pId = RPerson.Columns.pid.prefix("p") val pId = RPerson.Columns.pid.prefix("p")
val cPers = RContact.Columns.personId.prefix("c") val cPers = RContact.Columns.personId.prefix("c")
val cVal = RContact.Columns.value.prefix("c") val cVal = RContact.Columns.value.prefix("c")
val oId = ROrganization.Columns.oid.prefix("o")
val pOid = RPerson.Columns.oid.prefix("p")
val cols = RPerson.Columns.all.map(_.prefix("p")) ++ RContact.Columns.all val cols = RPerson.Columns.all.map(_.prefix("p")) ++
.map(_.prefix("c")) ROrganization.Columns.all.map(_.prefix("o")) ++
RContact.Columns.all.map(_.prefix("c"))
val from = RPerson.table ++ fr"p LEFT JOIN" ++ val from = RPerson.table ++ fr"p LEFT JOIN" ++
ROrganization.table ++ fr"o ON" ++ pOid.is(oId) ++ fr"LEFT JOIN" ++
RContact.table ++ fr"c ON" ++ cPers.is(pId) RContact.table ++ fr"c ON" ++ cPers.is(pId)
val q = Seq(pColl.is(coll)) ++ (query match { val q = Seq(pColl.is(coll)) ++ (query match {
@ -103,38 +107,44 @@ object QOrganization {
Seq.empty Seq.empty
}) })
(selectSimple(cols, from, and(q)) ++ orderBy(order(PC).f)) (selectSimple(cols, from, and(q)) ++ orderBy(order(PC).prefix("p").f))
.query[(RPerson, Option[RContact])] .query[(RPerson, Option[ROrganization], Option[RContact])]
.stream .stream
.groupAdjacentBy(_._1) .groupAdjacentBy(_._1)
.map({ case (ro, chunk) => .map({ case (rp, chunk) =>
val cs = chunk.toVector.flatMap(_._2) val cs = chunk.toVector.flatMap(_._3)
(ro, cs) val ro = chunk.map(_._2).head.flatten
(rp, ro, cs)
}) })
} }
def getPersonAndContact( def getPersonAndContact(
coll: Ident, coll: Ident,
persId: Ident persId: Ident
): ConnectionIO[Option[(RPerson, Vector[RContact])]] = { ): ConnectionIO[Option[(RPerson, Option[ROrganization], Vector[RContact])]] = {
val pColl = PC.cid.prefix("p") val pColl = PC.cid.prefix("p")
val pId = RPerson.Columns.pid.prefix("p") val pId = RPerson.Columns.pid.prefix("p")
val cPers = RContact.Columns.personId.prefix("c") val cPers = RContact.Columns.personId.prefix("c")
val oId = ROrganization.Columns.oid.prefix("o")
val pOid = RPerson.Columns.oid.prefix("p")
val cols = RPerson.Columns.all.map(_.prefix("p")) ++ RContact.Columns.all val cols = RPerson.Columns.all.map(_.prefix("p")) ++
.map(_.prefix("c")) ROrganization.Columns.all.map(_.prefix("o")) ++
RContact.Columns.all.map(_.prefix("c"))
val from = RPerson.table ++ fr"p LEFT JOIN" ++ val from = RPerson.table ++ fr"p LEFT JOIN" ++
ROrganization.table ++ fr"o ON" ++ pOid.is(oId) ++ fr"LEFT JOIN" ++
RContact.table ++ fr"c ON" ++ cPers.is(pId) RContact.table ++ fr"c ON" ++ cPers.is(pId)
val q = and(pColl.is(coll), pId.is(persId)) val q = and(pColl.is(coll), pId.is(persId))
selectSimple(cols, from, q) selectSimple(cols, from, q)
.query[(RPerson, Option[RContact])] .query[(RPerson, Option[ROrganization], Option[RContact])]
.stream .stream
.groupAdjacentBy(_._1) .groupAdjacentBy(_._1)
.map({ case (ro, chunk) => .map({ case (rp, chunk) =>
val cs = chunk.toVector.flatMap(_._2) val cs = chunk.toVector.flatMap(_._3)
(ro, cs) val ro = chunk.map(_._2).head.flatten
(rp, ro, cs)
}) })
.compile .compile
.last .last

View File

@ -1,6 +1,8 @@
package docspell.store.records package docspell.store.records
import cats.Eq import cats.Eq
import cats.data.NonEmptyList
import cats.effect._
import fs2.Stream import fs2.Stream
import docspell.common.{IdRef, _} import docspell.common.{IdRef, _}
@ -21,7 +23,8 @@ case class RPerson(
notes: Option[String], notes: Option[String],
concerning: Boolean, concerning: Boolean,
created: Timestamp, created: Timestamp,
updated: Timestamp updated: Timestamp,
oid: Option[Ident]
) {} ) {}
object RPerson { object RPerson {
@ -42,6 +45,7 @@ object RPerson {
val concerning = Column("concerning") val concerning = Column("concerning")
val created = Column("created") val created = Column("created")
val updated = Column("updated") val updated = Column("updated")
val oid = Column("oid")
val all = List( val all = List(
pid, pid,
cid, cid,
@ -53,7 +57,8 @@ object RPerson {
notes, notes,
concerning, concerning,
created, created,
updated updated,
oid
) )
} }
@ -63,7 +68,7 @@ object RPerson {
val sql = insertRow( val sql = insertRow(
table, table,
all, all,
fr"${v.pid},${v.cid},${v.name},${v.street},${v.zip},${v.city},${v.country},${v.notes},${v.concerning},${v.created},${v.updated}" fr"${v.pid},${v.cid},${v.name},${v.street},${v.zip},${v.city},${v.country},${v.notes},${v.concerning},${v.created},${v.updated},${v.oid}"
) )
sql.update.run sql.update.run
} }
@ -82,6 +87,7 @@ object RPerson {
country.setTo(v.country), country.setTo(v.country),
concerning.setTo(v.concerning), concerning.setTo(v.concerning),
notes.setTo(v.notes), notes.setTo(v.notes),
oid.setTo(v.oid),
updated.setTo(now) updated.setTo(now)
) )
) )
@ -163,4 +169,14 @@ object RPerson {
def delete(personId: Ident, coll: Ident): ConnectionIO[Int] = def delete(personId: Ident, coll: Ident): ConnectionIO[Int] =
deleteFrom(table, and(pid.is(personId), cid.is(coll))).update.run deleteFrom(table, and(pid.is(personId), cid.is(coll))).update.run
def findOrganization(ids: Set[Ident]): ConnectionIO[Vector[PersonRef]] = {
val cols = Seq(pid, name, oid)
NonEmptyList.fromList(ids.toList) match {
case Some(nel) =>
selectSimple(cols, table, pid.isIn(nel)).query[PersonRef].to[Vector]
case None =>
Sync[ConnectionIO].pure(Vector.empty)
}
}
} }

View File

@ -8,11 +8,9 @@ module Comp.CustomFieldTable exposing
) )
import Api.Model.CustomField exposing (CustomField) import Api.Model.CustomField exposing (CustomField)
import Api.Model.CustomFieldList exposing (CustomFieldList)
import Html exposing (..) import Html exposing (..)
import Html.Attributes exposing (..) import Html.Attributes exposing (..)
import Html.Events exposing (onClick) import Html.Events exposing (onClick)
import Util.Html
import Util.Time import Util.Time
@ -44,7 +42,7 @@ update msg model =
view : Model -> List CustomField -> Html Msg view : Model -> List CustomField -> Html Msg
view _ items = view _ items =
div [] div []
[ table [ class "ui very basic center aligned table" ] [ table [ class "ui very basic aligned table" ]
[ thead [] [ thead []
[ tr [] [ tr []
[ th [ class "collapsing" ] [] [ th [ class "collapsing" ] []

View File

@ -30,6 +30,7 @@ import Api.Model.Equipment exposing (Equipment)
import Api.Model.NewCustomField exposing (NewCustomField) import Api.Model.NewCustomField exposing (NewCustomField)
import Api.Model.Organization exposing (Organization) import Api.Model.Organization exposing (Organization)
import Api.Model.Person exposing (Person) import Api.Model.Person exposing (Person)
import Api.Model.ReferenceList exposing (ReferenceList)
import Api.Model.Tag exposing (Tag) import Api.Model.Tag exposing (Tag)
import Comp.CustomFieldForm import Comp.CustomFieldForm
import Comp.EquipmentForm import Comp.EquipmentForm
@ -134,7 +135,10 @@ editPerson flags persId pm =
, loading = True , loading = True
, result = Nothing , result = Nothing
} }
, Api.getPersonFull persId flags GetPersonResp , Cmd.batch
[ Api.getPersonFull persId flags GetPersonResp
, Api.getOrgLight flags GetOrgsResp
]
) )
@ -150,14 +154,18 @@ editEquip flags equipId em =
) )
initCorrPerson : String -> Comp.PersonForm.Model -> Model initCorrPerson : Flags -> String -> Comp.PersonForm.Model -> ( Model, Cmd Msg )
initCorrPerson itemId pm = initCorrPerson flags itemId pm =
init itemId (PMR pm) ( init itemId (PMR pm)
, Api.getOrgLight flags GetOrgsResp
)
initConcPerson : String -> Comp.PersonForm.Model -> Model initConcPerson : Flags -> String -> Comp.PersonForm.Model -> ( Model, Cmd Msg )
initConcPerson itemId pm = initConcPerson flags itemId pm =
init itemId (PMC pm) ( init itemId (PMC pm)
, Api.getOrgLight flags GetOrgsResp
)
initTag : String -> Comp.TagForm.Model -> Model initTag : String -> Comp.TagForm.Model -> Model
@ -198,6 +206,7 @@ type Msg
| GetOrgResp (Result Http.Error Organization) | GetOrgResp (Result Http.Error Organization)
| GetPersonResp (Result Http.Error Person) | GetPersonResp (Result Http.Error Person)
| GetEquipResp (Result Http.Error Equipment) | GetEquipResp (Result Http.Error Equipment)
| GetOrgsResp (Result Http.Error ReferenceList)
type Value type Value
@ -304,6 +313,43 @@ update flags msg model =
, Nothing , Nothing
) )
GetOrgsResp (Ok list) ->
case model.form of
PMC pm ->
let
( p_, c_ ) =
Comp.PersonForm.update flags (Comp.PersonForm.SetOrgs list.items) pm
in
( { model
| loading = False
, form = PMC p_
}
, Cmd.map PersonMsg c_
, Nothing
)
PMR pm ->
let
( p_, c_ ) =
Comp.PersonForm.update flags (Comp.PersonForm.SetOrgs list.items) pm
in
( { model
| loading = False
, form = PMR p_
}
, Cmd.map PersonMsg c_
, Nothing
)
_ ->
( { model | loading = False }, Cmd.none, Nothing )
GetOrgsResp (Err err) ->
( { model | loading = False, result = Just (BasicResult False (Util.Http.errorToString err)) }
, Cmd.none
, Nothing
)
GetEquipResp (Ok equip) -> GetEquipResp (Ok equip) ->
case model.form of case model.form of
EM em -> EM em ->

View File

@ -10,6 +10,7 @@ module Comp.Dropdown exposing
, makeSingleList , makeSingleList
, mkOption , mkOption
, notSelected , notSelected
, orgDropdown
, setMkOption , setMkOption
, update , update
, view , view
@ -19,6 +20,7 @@ module Comp.Dropdown exposing
{-| This needs to be rewritten from scratch! {-| This needs to be rewritten from scratch!
-} -}
import Api.Model.IdName exposing (IdName)
import Data.UiSettings exposing (UiSettings) import Data.UiSettings exposing (UiSettings)
import Html exposing (..) import Html exposing (..)
import Html.Attributes exposing (..) import Html.Attributes exposing (..)
@ -28,6 +30,17 @@ import Util.Html exposing (onKeyUp)
import Util.List import Util.List
orgDropdown : Model IdName
orgDropdown =
makeModel
{ multiple = False
, searchable = \n -> n > 0
, makeOption = \e -> { value = e.id, text = e.name, additional = "" }
, labelColor = \_ -> \_ -> ""
, placeholder = "Choose an organization"
}
type alias Option = type alias Option =
{ value : String { value : String
, text : String , text : String

View File

@ -47,10 +47,11 @@ update _ msg model =
view : Model -> Html Msg view : Model -> Html Msg
view model = view model =
table [ class "ui selectable table" ] table [ class "ui very basic aligned table" ]
[ thead [] [ thead []
[ tr [] [ tr []
[ th [] [ text "Name" ] [ th [ class "collapsing" ] []
, th [] [ text "Name" ]
] ]
] ]
, tbody [] , tbody []
@ -62,9 +63,18 @@ renderEquipmentLine : Model -> Equipment -> Html Msg
renderEquipmentLine model equip = renderEquipmentLine model equip =
tr tr
[ classList [ ( "active", model.selected == Just equip ) ] [ classList [ ( "active", model.selected == Just equip ) ]
, onClick (Select equip)
] ]
[ td [] [ td [ class "collapsing" ]
[ a
[ href "#"
, class "ui basic small blue label"
, onClick (Select equip)
]
[ i [ class "edit icon" ] []
, text "Edit"
]
]
, td []
[ text equip.name [ text equip.name
] ]
] ]

View File

@ -43,15 +43,15 @@ update msg model =
view : Model -> List FolderItem -> Html Msg view : Model -> List FolderItem -> Html Msg
view _ items = view _ items =
div [] div []
[ table [ class "ui very basic center aligned table" ] [ table [ class "ui very basic aligned table" ]
[ thead [] [ thead []
[ tr [] [ tr []
[ th [ class "collapsing" ] [] [ th [ class "collapsing" ] []
, th [] [ text "Name" ] , th [] [ text "Name" ]
, th [] [ text "Owner" ] , th [] [ text "Owner" ]
, th [] [ text "Owner or Member" ] , th [ class "collapsing" ] [ text "Owner or Member" ]
, th [] [ text "#Member" ] , th [ class "collapsing" ] [ text "#Member" ]
, th [] [ text "Created" ] , th [ class "collapsing" ] [ text "Created" ]
] ]
] ]
, tbody [] , tbody []
@ -79,14 +79,14 @@ viewItem item =
, td [] , td []
[ text item.owner.name [ text item.owner.name
] ]
, td [] , td [ class "center aligned" ]
[ Util.Html.checkbox item.isMember [ Util.Html.checkbox item.isMember
] ]
, td [] , td [ class "center aligned" ]
[ String.fromInt item.memberCount [ String.fromInt item.memberCount
|> text |> text
] ]
, td [] , td [ class "center aligned" ]
[ Util.Time.formatDateShort item.created [ Util.Time.formatDateShort item.created
|> text |> text
] ]

View File

@ -7,6 +7,7 @@ module Comp.ItemDetail.Model exposing
, UpdateResult , UpdateResult
, emptyModel , emptyModel
, isEditNotes , isEditNotes
, personMatchesOrg
, resultModel , resultModel
, resultModelCmd , resultModelCmd
, resultModelCmdSub , resultModelCmdSub
@ -20,6 +21,7 @@ import Api.Model.FolderList exposing (FolderList)
import Api.Model.IdName exposing (IdName) import Api.Model.IdName exposing (IdName)
import Api.Model.ItemDetail exposing (ItemDetail) import Api.Model.ItemDetail exposing (ItemDetail)
import Api.Model.ItemProposals exposing (ItemProposals) import Api.Model.ItemProposals exposing (ItemProposals)
import Api.Model.Person exposing (Person)
import Api.Model.PersonList exposing (PersonList) import Api.Model.PersonList exposing (PersonList)
import Api.Model.ReferenceList exposing (ReferenceList) import Api.Model.ReferenceList exposing (ReferenceList)
import Api.Model.SentMails exposing (SentMails) import Api.Model.SentMails exposing (SentMails)
@ -100,6 +102,7 @@ type alias Model =
, customFieldSavingIcon : Dict String String , customFieldSavingIcon : Dict String String
, customFieldThrottle : Throttle Msg , customFieldThrottle : Throttle Msg
, allTags : List Tag , allTags : List Tag
, allPersons : Dict String Person
} }
@ -205,6 +208,7 @@ emptyModel =
, customFieldSavingIcon = Dict.empty , customFieldSavingIcon = Dict.empty
, customFieldThrottle = Throttle.create 1 , customFieldThrottle = Throttle.create 1
, allTags = [] , allTags = []
, allPersons = Dict.empty
} }
@ -322,3 +326,19 @@ resultModelCmd ( model, cmd ) =
resultModelCmdSub : ( Model, Cmd Msg, Sub Msg ) -> UpdateResult resultModelCmdSub : ( Model, Cmd Msg, Sub Msg ) -> UpdateResult
resultModelCmdSub ( model, cmd, sub ) = resultModelCmdSub ( model, cmd, sub ) =
UpdateResult model cmd sub Comp.LinkTarget.LinkNone UpdateResult model cmd sub Comp.LinkTarget.LinkNone
personMatchesOrg : Model -> Bool
personMatchesOrg model =
let
org =
Comp.Dropdown.getSelected model.corrOrgModel
|> List.head
persOrg =
Comp.Dropdown.getSelected model.corrPersonModel
|> List.head
|> Maybe.andThen (\idref -> Dict.get idref.id model.allPersons)
|> Maybe.andThen .organization
in
org == Nothing || org == persOrg

View File

@ -252,6 +252,7 @@ update key flags inav settings msg model =
, getOptions flags , getOptions flags
, proposalCmd , proposalCmd
, Api.getSentMails flags item.id SentMailsResp , Api.getSentMails flags item.id SentMailsResp
, Api.getPersons flags "" GetPersonResp
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags) , Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
] ]
, sub = , sub =
@ -347,12 +348,14 @@ update key flags inav settings msg model =
( m2, c2 ) = ( m2, c2 ) =
Comp.Dropdown.update m model.corrOrgModel Comp.Dropdown.update m model.corrOrgModel
newModel =
{ model | corrOrgModel = m2 }
idref = idref =
Comp.Dropdown.getSelected m2 |> List.head Comp.Dropdown.getSelected m2 |> List.head
newModel =
{ model
| corrOrgModel = m2
}
save = save =
if isDropdownChangeMsg m then if isDropdownChangeMsg m then
setCorrOrg flags newModel idref setCorrOrg flags newModel idref
@ -609,11 +612,46 @@ update key flags inav settings msg model =
( conc, corr ) = ( conc, corr ) =
List.partition .concerning ps.items List.partition .concerning ps.items
personDict =
List.map (\p -> ( p.id, p )) ps.items
|> Dict.fromList
corrOrg =
Comp.Dropdown.getSelected model.corrOrgModel
|> List.head
personFilter =
case corrOrg of
Just n ->
\p -> p.organization == Just n
Nothing ->
\_ -> True
concRefs = concRefs =
List.map (\e -> IdName e.id e.name) conc List.map (\e -> IdName e.id e.name) conc
corrRefs = corrRefs =
List.map (\e -> IdName e.id e.name) corr List.filter personFilter corr
|> List.map (\e -> IdName e.id e.name)
mkPersonOption idref =
let
org =
Dict.get idref.id personDict
|> Maybe.andThen .organization
|> Maybe.map .name
|> Maybe.map (Util.String.ellipsis 15)
|> Maybe.withDefault ""
in
Comp.Dropdown.Option idref.id idref.name org
model_ =
{ model
| corrPersonModel = Comp.Dropdown.setMkOption mkPersonOption model.corrPersonModel
, concPersonModel = Comp.Dropdown.setMkOption mkPersonOption model.concPersonModel
, allPersons = personDict
}
res1 = res1 =
update key update key
@ -621,7 +659,7 @@ update key flags inav settings msg model =
inav inav
settings settings
(CorrPersonMsg (Comp.Dropdown.SetOptions corrRefs)) (CorrPersonMsg (Comp.Dropdown.SetOptions corrRefs))
model model_
res2 = res2 =
update key update key
@ -1113,26 +1151,30 @@ update key flags inav settings msg model =
resultModel model resultModel model
StartCorrPersonModal -> StartCorrPersonModal ->
resultModel let
{ model ( pm, pc ) =
| modalEdit = Comp.DetailEdit.initCorrPerson
Just flags
(Comp.DetailEdit.initCorrPerson model.item.id
model.item.id Comp.PersonForm.emptyModel
Comp.PersonForm.emptyModel in
) resultModelCmd
} ( { model | modalEdit = Just pm }
, Cmd.map ModalEditMsg pc
)
StartConcPersonModal -> StartConcPersonModal ->
resultModel let
{ model ( p, c ) =
| modalEdit = Comp.DetailEdit.initConcPerson
Just flags
(Comp.DetailEdit.initConcPerson model.item.id
model.item.id Comp.PersonForm.emptyModel
Comp.PersonForm.emptyModel in
) resultModelCmd
} ( { model | modalEdit = Just p }
, Cmd.map ModalEditMsg c
)
StartEditPersonModal pm -> StartEditPersonModal pm ->
let let

View File

@ -10,14 +10,20 @@ import Comp.DetailEdit
import Comp.Dropdown import Comp.Dropdown
import Comp.Dropzone import Comp.Dropzone
import Comp.ItemDetail.AttachmentTabMenu import Comp.ItemDetail.AttachmentTabMenu
import Comp.ItemDetail.Model exposing (Model, Msg(..), NotesField(..), SaveNameState(..)) import Comp.ItemDetail.Model
exposing
( Model
, Msg(..)
, NotesField(..)
, SaveNameState(..)
, personMatchesOrg
)
import Comp.ItemMail import Comp.ItemMail
import Comp.KeyInput import Comp.KeyInput
import Comp.LinkTarget import Comp.LinkTarget
import Comp.MarkdownInput import Comp.MarkdownInput
import Comp.SentMails import Comp.SentMails
import Comp.YesNoDimmer import Comp.YesNoDimmer
import Data.CustomFieldType
import Data.Direction import Data.Direction
import Data.Fields import Data.Fields
import Data.Icons as Icons import Data.Icons as Icons
@ -883,6 +889,15 @@ item visible. This message will disappear then.
] ]
, Html.map CorrPersonMsg (Comp.Dropdown.view settings model.corrPersonModel) , Html.map CorrPersonMsg (Comp.Dropdown.view settings model.corrPersonModel)
, renderCorrPersonSuggestions model , renderCorrPersonSuggestions model
, div
[ classList
[ ( "ui warning message", True )
, ( "invisible hidden", personMatchesOrg model )
]
]
[ i [ class "info icon" ] []
, text "The selected person doesn't belong to the selected organization."
]
] ]
, optional [ Data.Fields.ConcPerson, Data.Fields.ConcEquip ] <| , optional [ Data.Fields.ConcPerson, Data.Fields.ConcEquip ] <|
h4 [ class "ui dividing header" ] h4 [ class "ui dividing header" ]

View File

@ -16,14 +16,14 @@ import Util.Contact
type alias Model = type alias Model =
{ equips : List Organization { orgs : List Organization
, selected : Maybe Organization , selected : Maybe Organization
} }
emptyModel : Model emptyModel : Model
emptyModel = emptyModel =
{ equips = [] { orgs = []
, selected = Nothing , selected = Nothing
} }
@ -38,7 +38,7 @@ update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
update _ msg model = update _ msg model =
case msg of case msg of
SetOrgs list -> SetOrgs list ->
( { model | equips = list, selected = Nothing }, Cmd.none ) ( { model | orgs = list, selected = Nothing }, Cmd.none )
Select equip -> Select equip ->
( { model | selected = Just equip }, Cmd.none ) ( { model | selected = Just equip }, Cmd.none )
@ -49,16 +49,17 @@ update _ msg model =
view : Model -> Html Msg view : Model -> Html Msg
view model = view model =
table [ class "ui selectable table" ] table [ class "ui very basic aligned table" ]
[ thead [] [ thead []
[ tr [] [ tr []
[ th [ class "collapsing" ] [ text "Name" ] [ th [ class "collapsing" ] []
, th [ class "collapsing" ] [ text "Name" ]
, th [] [ text "Address" ] , th [] [ text "Address" ]
, th [] [ text "Contact" ] , th [] [ text "Contact" ]
] ]
] ]
, tbody [] , tbody []
(List.map (renderOrgLine model) model.equips) (List.map (renderOrgLine model) model.orgs)
] ]
@ -66,9 +67,18 @@ renderOrgLine : Model -> Organization -> Html Msg
renderOrgLine model org = renderOrgLine model org =
tr tr
[ classList [ ( "active", model.selected == Just org ) ] [ classList [ ( "active", model.selected == Just org ) ]
, onClick (Select org)
] ]
[ td [ class "collapsing" ] [ td [ class "collapsing" ]
[ a
[ href "#"
, class "ui basic small blue label"
, onClick (Select org)
]
[ i [ class "edit icon" ] []
, text "Edit"
]
]
, td [ class "collapsing" ]
[ text org.name [ text org.name
] ]
, td [] , td []

View File

@ -9,9 +9,11 @@ module Comp.PersonForm exposing
, view1 , view1
) )
import Api.Model.IdName exposing (IdName)
import Api.Model.Person exposing (Person) import Api.Model.Person exposing (Person)
import Comp.AddressForm import Comp.AddressForm
import Comp.ContactField import Comp.ContactField
import Comp.Dropdown
import Data.Flags exposing (Flags) import Data.Flags exposing (Flags)
import Data.UiSettings exposing (UiSettings) import Data.UiSettings exposing (UiSettings)
import Html exposing (..) import Html exposing (..)
@ -20,23 +22,25 @@ import Html.Events exposing (onCheck, onInput)
type alias Model = type alias Model =
{ org : Person { person : Person
, name : String , name : String
, addressModel : Comp.AddressForm.Model , addressModel : Comp.AddressForm.Model
, contactModel : Comp.ContactField.Model , contactModel : Comp.ContactField.Model
, notes : Maybe String , notes : Maybe String
, concerning : Bool , concerning : Bool
, orgModel : Comp.Dropdown.Model IdName
} }
emptyModel : Model emptyModel : Model
emptyModel = emptyModel =
{ org = Api.Model.Person.empty { person = Api.Model.Person.empty
, name = "" , name = ""
, addressModel = Comp.AddressForm.emptyModel , addressModel = Comp.AddressForm.emptyModel
, contactModel = Comp.ContactField.emptyModel , contactModel = Comp.ContactField.emptyModel
, notes = Nothing , notes = Nothing
, concerning = False , concerning = False
, orgModel = Comp.Dropdown.orgDropdown
} }
@ -48,15 +52,20 @@ isValid model =
getPerson : Model -> Person getPerson : Model -> Person
getPerson model = getPerson model =
let let
o = person =
model.org model.person
org =
Comp.Dropdown.getSelected model.orgModel
|> List.head
in in
{ o { person
| name = model.name | name = model.name
, address = Comp.AddressForm.getAddress model.addressModel , address = Comp.AddressForm.getAddress model.addressModel
, contacts = Comp.ContactField.getContacts model.contactModel , contacts = Comp.ContactField.getContacts model.contactModel
, notes = model.notes , notes = model.notes
, concerning = model.concerning , concerning = model.concerning
, organization = org
} }
@ -67,6 +76,8 @@ type Msg
| ContactMsg Comp.ContactField.Msg | ContactMsg Comp.ContactField.Msg
| SetNotes String | SetNotes String
| SetConcerning Bool | SetConcerning Bool
| SetOrgs (List IdName)
| OrgDropdownMsg (Comp.Dropdown.Msg IdName)
update : Flags -> Msg -> Model -> ( Model, Cmd Msg ) update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
@ -79,16 +90,32 @@ update flags msg model =
( m2, c2 ) = ( m2, c2 ) =
update flags (ContactMsg (Comp.ContactField.SetItems t.contacts)) m1 update flags (ContactMsg (Comp.ContactField.SetItems t.contacts)) m1
( m3, c3 ) =
update flags
(OrgDropdownMsg
(Comp.Dropdown.SetSelection
(List.filterMap identity [ t.organization ])
)
)
m2
in in
( { m2 ( { m3
| org = t | person = t
, name = t.name , name = t.name
, notes = t.notes , notes = t.notes
, concerning = t.concerning , concerning = t.concerning
} }
, Cmd.batch [ c1, c2 ] , Cmd.batch [ c1, c2, c3 ]
) )
SetOrgs orgs ->
let
opts =
Comp.Dropdown.SetOptions orgs
in
update flags (OrgDropdownMsg opts) model
AddressMsg am -> AddressMsg am ->
let let
( m1, c1 ) = ( m1, c1 ) =
@ -121,6 +148,15 @@ update flags msg model =
SetConcerning _ -> SetConcerning _ ->
( { model | concerning = not model.concerning }, Cmd.none ) ( { model | concerning = not model.concerning }, Cmd.none )
OrgDropdownMsg lm ->
let
( dm_, cmd_ ) =
Comp.Dropdown.update lm model.orgModel
in
( { model | orgModel = dm_ }
, Cmd.map OrgDropdownMsg cmd_
)
view : UiSettings -> Model -> Html Msg view : UiSettings -> Model -> Html Msg
view settings model = view settings model =
@ -156,6 +192,10 @@ view1 settings compact model =
, label [] [ text "Use for concerning person suggestion only" ] , label [] [ text "Use for concerning person suggestion only" ]
] ]
] ]
, div [ class "field" ]
[ label [] [ text "Organization" ]
, Html.map OrgDropdownMsg (Comp.Dropdown.view settings model.orgModel)
]
, h3 [ class "ui dividing header" ] , h3 [ class "ui dividing header" ]
[ text "Address" [ text "Address"
] ]

View File

@ -10,6 +10,7 @@ import Api
import Api.Model.BasicResult exposing (BasicResult) import Api.Model.BasicResult exposing (BasicResult)
import Api.Model.Person import Api.Model.Person
import Api.Model.PersonList exposing (PersonList) import Api.Model.PersonList exposing (PersonList)
import Api.Model.ReferenceList exposing (ReferenceList)
import Comp.PersonForm import Comp.PersonForm
import Comp.PersonTable import Comp.PersonTable
import Comp.YesNoDimmer import Comp.YesNoDimmer
@ -28,7 +29,7 @@ type alias Model =
, formModel : Comp.PersonForm.Model , formModel : Comp.PersonForm.Model
, viewMode : ViewMode , viewMode : ViewMode
, formError : Maybe String , formError : Maybe String
, loading : Bool , loading : Int
, deleteConfirm : Comp.YesNoDimmer.Model , deleteConfirm : Comp.YesNoDimmer.Model
, query : String , query : String
} }
@ -45,7 +46,7 @@ emptyModel =
, formModel = Comp.PersonForm.emptyModel , formModel = Comp.PersonForm.emptyModel
, viewMode = Table , viewMode = Table
, formError = Nothing , formError = Nothing
, loading = False , loading = 0
, deleteConfirm = Comp.YesNoDimmer.emptyModel , deleteConfirm = Comp.YesNoDimmer.emptyModel
, query = "" , query = ""
} }
@ -63,6 +64,7 @@ type Msg
| YesNoMsg Comp.YesNoDimmer.Msg | YesNoMsg Comp.YesNoDimmer.Msg
| RequestDelete | RequestDelete
| SetQuery String | SetQuery String
| GetOrgResp (Result Http.Error ReferenceList)
update : Flags -> Msg -> Model -> ( Model, Cmd Msg ) update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
@ -105,17 +107,35 @@ update flags msg model =
( { model | formModel = m2 }, Cmd.map FormMsg c2 ) ( { model | formModel = m2 }, Cmd.map FormMsg c2 )
LoadPersons -> LoadPersons ->
( { model | loading = True }, Api.getPersons flags model.query PersonResp ) ( { model | loading = model.loading + 2 }
, Cmd.batch
[ Api.getPersons flags model.query PersonResp
, Api.getOrgLight flags GetOrgResp
]
)
PersonResp (Ok orgs) -> PersonResp (Ok persons) ->
let let
m2 = m2 =
{ model | viewMode = Table, loading = False } { model
| viewMode = Table
, loading = Basics.max 0 (model.loading - 1)
}
in in
update flags (TableMsg (Comp.PersonTable.SetPersons orgs.items)) m2 update flags (TableMsg (Comp.PersonTable.SetPersons persons.items)) m2
PersonResp (Err _) -> PersonResp (Err _) ->
( { model | loading = False }, Cmd.none ) ( { model | loading = Basics.max 0 (model.loading - 1) }, Cmd.none )
GetOrgResp (Ok list) ->
let
m2 =
{ model | loading = Basics.max 0 (model.loading - 1) }
in
update flags (FormMsg (Comp.PersonForm.SetOrgs list.items)) m2
GetOrgResp (Err _) ->
( { model | loading = Basics.max 0 (model.loading - 1) }, Cmd.none )
SetViewMode m -> SetViewMode m ->
let let
@ -148,7 +168,9 @@ update flags msg model =
Comp.PersonForm.isValid model.formModel Comp.PersonForm.isValid model.formModel
in in
if valid then if valid then
( { model | loading = True }, Api.postPerson flags person SubmitResp ) ( { model | loading = model.loading + 1 }
, Api.postPerson flags person SubmitResp
)
else else
( { model | formError = Just "Please correct the errors in the form." }, Cmd.none ) ( { model | formError = Just "Please correct the errors in the form." }, Cmd.none )
@ -162,13 +184,23 @@ update flags msg model =
( m3, c3 ) = ( m3, c3 ) =
update flags LoadPersons m2 update flags LoadPersons m2
in in
( { m3 | loading = False }, Cmd.batch [ c2, c3 ] ) ( { m3 | loading = Basics.max 0 (model.loading - 1) }, Cmd.batch [ c2, c3 ] )
else else
( { model | formError = Just res.message, loading = False }, Cmd.none ) ( { model
| formError = Just res.message
, loading = Basics.max 0 (model.loading - 1)
}
, Cmd.none
)
SubmitResp (Err err) -> SubmitResp (Err err) ->
( { model | formError = Just (Util.Http.errorToString err), loading = False }, Cmd.none ) ( { model
| formError = Just (Util.Http.errorToString err)
, loading = Basics.max 0 (model.loading - 1)
}
, Cmd.none
)
RequestDelete -> RequestDelete ->
update flags (YesNoMsg Comp.YesNoDimmer.activate) model update flags (YesNoMsg Comp.YesNoDimmer.activate) model
@ -198,6 +230,11 @@ update flags msg model =
( m, Api.getPersons flags str PersonResp ) ( m, Api.getPersons flags str PersonResp )
isLoading : Model -> Bool
isLoading model =
model.loading /= 0
view : UiSettings -> Model -> Html Msg view : UiSettings -> Model -> Html Msg
view settings model = view settings model =
if model.viewMode == Table then if model.viewMode == Table then
@ -241,7 +278,7 @@ viewTable model =
, div , div
[ classList [ classList
[ ( "ui dimmer", True ) [ ( "ui dimmer", True )
, ( "active", model.loading ) , ( "active", isLoading model )
] ]
] ]
[ div [ class "ui loader" ] [] [ div [ class "ui loader" ] []
@ -253,9 +290,9 @@ viewForm : UiSettings -> Model -> Html Msg
viewForm settings model = viewForm settings model =
let let
newPerson = newPerson =
model.formModel.org.id == "" model.formModel.person.id == ""
in in
Html.form [ class "ui segment", onSubmit Submit ] div [ class "ui segment" ]
[ Html.map YesNoMsg (Comp.YesNoDimmer.view model.deleteConfirm) [ Html.map YesNoMsg (Comp.YesNoDimmer.view model.deleteConfirm)
, if newPerson then , if newPerson then
h3 [ class "ui dividing header" ] h3 [ class "ui dividing header" ]
@ -264,10 +301,10 @@ viewForm settings model =
else else
h3 [ class "ui dividing header" ] h3 [ class "ui dividing header" ]
[ text ("Edit person: " ++ model.formModel.org.name) [ text ("Edit person: " ++ model.formModel.person.name)
, div [ class "sub header" ] , div [ class "sub header" ]
[ text "Id: " [ text "Id: "
, text model.formModel.org.id , text model.formModel.person.id
] ]
] ]
, Html.map FormMsg (Comp.PersonForm.view settings model.formModel) , Html.map FormMsg (Comp.PersonForm.view settings model.formModel)
@ -280,14 +317,25 @@ viewForm settings model =
[ Maybe.withDefault "" model.formError |> text [ Maybe.withDefault "" model.formError |> text
] ]
, div [ class "ui horizontal divider" ] [] , div [ class "ui horizontal divider" ] []
, button [ class "ui primary button", type_ "submit" ] , button
[ class "ui primary button"
, onClick Submit
]
[ text "Submit" [ text "Submit"
] ]
, a [ class "ui secondary button", onClick (SetViewMode Table), href "" ] , a
[ class "ui secondary button"
, onClick (SetViewMode Table)
, href ""
]
[ text "Cancel" [ text "Cancel"
] ]
, if not newPerson then , if not newPerson then
a [ class "ui right floated red button", href "", onClick RequestDelete ] a
[ class "ui right floated red button"
, href ""
, onClick RequestDelete
]
[ text "Delete" ] [ text "Delete" ]
else else
@ -295,7 +343,7 @@ viewForm settings model =
, div , div
[ classList [ classList
[ ( "ui dimmer", True ) [ ( "ui dimmer", True )
, ( "active", model.loading ) , ( "active", isLoading model )
] ]
] ]
[ div [ class "ui loader" ] [] [ div [ class "ui loader" ] []

View File

@ -49,11 +49,13 @@ update _ msg model =
view : Model -> Html Msg view : Model -> Html Msg
view model = view model =
table [ class "ui selectable table" ] table [ class "ui very basic aligned table" ]
[ thead [] [ thead []
[ tr [] [ tr []
[ th [ class "collapsing" ] [ text "Name" ] [ th [ class "collapsing" ] []
, th [ class "collapsing" ] [ text "Concerning" ] , th [ class "collapsing center aligned" ] [ text "Concerning" ]
, th [] [ text "Name" ]
, th [] [ text "Organization" ]
, th [] [ text "Address" ] , th [] [ text "Address" ]
, th [] [ text "Contact" ] , th [] [ text "Contact" ]
] ]
@ -67,18 +69,32 @@ renderPersonLine : Model -> Person -> Html Msg
renderPersonLine model person = renderPersonLine model person =
tr tr
[ classList [ ( "active", model.selected == Just person ) ] [ classList [ ( "active", model.selected == Just person ) ]
, onClick (Select person)
] ]
[ td [ class "collapsing" ] [ td [ class "collapsing" ]
[ text person.name [ a
[ href "#"
, class "ui basic small blue label"
, onClick (Select person)
]
[ i [ class "edit icon" ] []
, text "Edit"
]
] ]
, td [ class "collapsing" ] , td [ class "center aligned" ]
[ if person.concerning then [ if person.concerning then
i [ class "check square outline icon" ] [] i [ class "check square outline icon" ] []
else else
i [ class "minus square outline icon" ] [] i [ class "minus square outline icon" ] []
] ]
, td []
[ text person.name
]
, td []
[ Maybe.map .name person.organization
|> Maybe.withDefault "-"
|> text
]
, td [] , td []
[ Util.Address.toString person.address |> text [ Util.Address.toString person.address |> text
] ]

View File

@ -93,13 +93,7 @@ init =
, selected = Nothing , selected = Nothing
} }
, orgModel = , orgModel =
Comp.Dropdown.makeModel Comp.Dropdown.orgDropdown
{ multiple = False
, searchable = \n -> n > 0
, makeOption = \e -> { value = e.id, text = e.name, additional = "" }
, labelColor = \_ -> \_ -> ""
, placeholder = "Choose an organization"
}
, corrPersonModel = , corrPersonModel =
Comp.Dropdown.makeSingle Comp.Dropdown.makeSingle
{ makeOption = \e -> { value = e.id, text = e.name, additional = "" } { makeOption = \e -> { value = e.id, text = e.name, additional = "" }

View File

@ -47,10 +47,11 @@ update _ msg model =
view : Model -> Html Msg view : Model -> Html Msg
view model = view model =
table [ class "ui selectable table" ] table [ class "ui very basic aligned table" ]
[ thead [] [ thead []
[ tr [] [ tr []
[ th [] [ text "Name" ] [ th [ class "collapsing" ] []
, th [ class "eight wide" ] [ text "Name" ]
, th [] [ text "Category" ] , th [] [ text "Category" ]
] ]
] ]
@ -63,9 +64,18 @@ renderTagLine : Model -> Tag -> Html Msg
renderTagLine model tag = renderTagLine model tag =
tr tr
[ classList [ ( "active", model.selected == Just tag ) ] [ classList [ ( "active", model.selected == Just tag ) ]
, onClick (Select tag)
] ]
[ td [] [ td [ class "collapsing" ]
[ a
[ href "#"
, class "ui basic small blue label"
, onClick (Select tag)
]
[ i [ class "edit icon" ] []
, text "Edit"
]
]
, td []
[ text tag.name [ text tag.name
] ]
, td [] , td []