mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 19:09:32 +00:00
Merge pull request #1037 from eikek/small-list-improvements
Allow to specify ordering when retrieving meta data
This commit is contained in:
commit
3520a2ec26
@ -7,12 +7,13 @@
|
|||||||
package docspell.backend.ops
|
package docspell.backend.ops
|
||||||
|
|
||||||
import cats.data.EitherT
|
import cats.data.EitherT
|
||||||
import cats.data.NonEmptyList
|
|
||||||
import cats.data.OptionT
|
import cats.data.OptionT
|
||||||
|
import cats.data.{NonEmptyList => Nel}
|
||||||
import cats.effect._
|
import cats.effect._
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
import docspell.backend.ops.OCustomFields.CustomFieldData
|
import docspell.backend.ops.OCustomFields.CustomFieldData
|
||||||
|
import docspell.backend.ops.OCustomFields.CustomFieldOrder
|
||||||
import docspell.backend.ops.OCustomFields.FieldValue
|
import docspell.backend.ops.OCustomFields.FieldValue
|
||||||
import docspell.backend.ops.OCustomFields.NewCustomField
|
import docspell.backend.ops.OCustomFields.NewCustomField
|
||||||
import docspell.backend.ops.OCustomFields.RemoveValue
|
import docspell.backend.ops.OCustomFields.RemoveValue
|
||||||
@ -33,7 +34,11 @@ import org.log4s.getLogger
|
|||||||
trait OCustomFields[F[_]] {
|
trait OCustomFields[F[_]] {
|
||||||
|
|
||||||
/** Find all fields using an optional query on the name and label */
|
/** Find all fields using an optional query on the name and label */
|
||||||
def findAll(coll: Ident, nameQuery: Option[String]): F[Vector[CustomFieldData]]
|
def findAll(
|
||||||
|
coll: Ident,
|
||||||
|
nameQuery: Option[String],
|
||||||
|
order: CustomFieldOrder
|
||||||
|
): F[Vector[CustomFieldData]]
|
||||||
|
|
||||||
/** Find one field by its id */
|
/** Find one field by its id */
|
||||||
def findById(coll: Ident, fieldId: Ident): F[Option[CustomFieldData]]
|
def findById(coll: Ident, fieldId: Ident): F[Option[CustomFieldData]]
|
||||||
@ -50,13 +55,13 @@ trait OCustomFields[F[_]] {
|
|||||||
/** Sets a value given a field an an item. Existing values are overwritten. */
|
/** Sets a value given a field an an item. Existing values are overwritten. */
|
||||||
def setValue(item: Ident, value: SetValue): F[SetValueResult]
|
def setValue(item: Ident, value: SetValue): F[SetValueResult]
|
||||||
|
|
||||||
def setValueMultiple(items: NonEmptyList[Ident], value: SetValue): F[SetValueResult]
|
def setValueMultiple(items: Nel[Ident], value: SetValue): F[SetValueResult]
|
||||||
|
|
||||||
/** Deletes a value for a given field an item. */
|
/** Deletes a value for a given field an item. */
|
||||||
def deleteValue(in: RemoveValue): F[UpdateResult]
|
def deleteValue(in: RemoveValue): F[UpdateResult]
|
||||||
|
|
||||||
/** Finds all values to the given items */
|
/** Finds all values to the given items */
|
||||||
def findAllValues(itemIds: NonEmptyList[Ident]): F[List[FieldValue]]
|
def findAllValues(itemIds: Nel[Ident]): F[List[FieldValue]]
|
||||||
}
|
}
|
||||||
|
|
||||||
object OCustomFields {
|
object OCustomFields {
|
||||||
@ -96,10 +101,48 @@ object OCustomFields {
|
|||||||
|
|
||||||
case class RemoveValue(
|
case class RemoveValue(
|
||||||
field: Ident,
|
field: Ident,
|
||||||
item: NonEmptyList[Ident],
|
item: Nel[Ident],
|
||||||
collective: Ident
|
collective: Ident
|
||||||
)
|
)
|
||||||
|
|
||||||
|
sealed trait CustomFieldOrder
|
||||||
|
object CustomFieldOrder {
|
||||||
|
import docspell.store.qb.DSL._
|
||||||
|
|
||||||
|
final case object NameAsc extends CustomFieldOrder
|
||||||
|
final case object NameDesc extends CustomFieldOrder
|
||||||
|
final case object LabelAsc extends CustomFieldOrder
|
||||||
|
final case object LabelDesc extends CustomFieldOrder
|
||||||
|
final case object TypeAsc extends CustomFieldOrder
|
||||||
|
final case object TypeDesc extends CustomFieldOrder
|
||||||
|
|
||||||
|
def parse(str: String): Either[String, CustomFieldOrder] =
|
||||||
|
str.toLowerCase match {
|
||||||
|
case "name" => Right(NameAsc)
|
||||||
|
case "-name" => Right(NameDesc)
|
||||||
|
case "label" => Right(LabelAsc)
|
||||||
|
case "-label" => Right(LabelDesc)
|
||||||
|
case "type" => Right(TypeAsc)
|
||||||
|
case "-type" => Right(TypeDesc)
|
||||||
|
case _ => Left(s"Unknown sort property for custom field: $str")
|
||||||
|
}
|
||||||
|
|
||||||
|
def parseOrDefault(str: String): CustomFieldOrder =
|
||||||
|
parse(str).toOption.getOrElse(NameAsc)
|
||||||
|
|
||||||
|
private[ops] def apply(
|
||||||
|
order: CustomFieldOrder
|
||||||
|
)(field: RCustomField.Table) =
|
||||||
|
order match {
|
||||||
|
case NameAsc => Nel.of(field.name.asc)
|
||||||
|
case NameDesc => Nel.of(field.name.desc)
|
||||||
|
case LabelAsc => Nel.of(coalesce(field.label.s, field.name.s).asc)
|
||||||
|
case LabelDesc => Nel.of(coalesce(field.label.s, field.name.s).desc)
|
||||||
|
case TypeAsc => Nel.of(field.ftype.asc, field.name.asc)
|
||||||
|
case TypeDesc => Nel.of(field.ftype.desc, field.name.desc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def apply[F[_]: Async](
|
def apply[F[_]: Async](
|
||||||
store: Store[F]
|
store: Store[F]
|
||||||
): Resource[F, OCustomFields[F]] =
|
): Resource[F, OCustomFields[F]] =
|
||||||
@ -107,14 +150,19 @@ object OCustomFields {
|
|||||||
|
|
||||||
private[this] val logger = Logger.log4s[ConnectionIO](getLogger)
|
private[this] val logger = Logger.log4s[ConnectionIO](getLogger)
|
||||||
|
|
||||||
def findAllValues(itemIds: NonEmptyList[Ident]): F[List[FieldValue]] =
|
def findAllValues(itemIds: Nel[Ident]): F[List[FieldValue]] =
|
||||||
store.transact(QCustomField.findAllValues(itemIds))
|
store.transact(QCustomField.findAllValues(itemIds))
|
||||||
|
|
||||||
def findAll(coll: Ident, nameQuery: Option[String]): F[Vector[CustomFieldData]] =
|
def findAll(
|
||||||
|
coll: Ident,
|
||||||
|
nameQuery: Option[String],
|
||||||
|
order: CustomFieldOrder
|
||||||
|
): F[Vector[CustomFieldData]] =
|
||||||
store.transact(
|
store.transact(
|
||||||
QCustomField.findAllLike(
|
QCustomField.findAllLike(
|
||||||
coll,
|
coll,
|
||||||
nameQuery.map(WildcardString.apply).flatMap(_.both)
|
nameQuery.map(WildcardString.apply).flatMap(_.both),
|
||||||
|
CustomFieldOrder(order)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -149,10 +197,10 @@ object OCustomFields {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def setValue(item: Ident, value: SetValue): F[SetValueResult] =
|
def setValue(item: Ident, value: SetValue): F[SetValueResult] =
|
||||||
setValueMultiple(NonEmptyList.of(item), value)
|
setValueMultiple(Nel.of(item), value)
|
||||||
|
|
||||||
def setValueMultiple(
|
def setValueMultiple(
|
||||||
items: NonEmptyList[Ident],
|
items: Nel[Ident],
|
||||||
value: SetValue
|
value: SetValue
|
||||||
): F[SetValueResult] =
|
): F[SetValueResult] =
|
||||||
(for {
|
(for {
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
package docspell.backend.ops
|
package docspell.backend.ops
|
||||||
|
|
||||||
|
import cats.data.NonEmptyList
|
||||||
import cats.effect.{Async, Resource}
|
import cats.effect.{Async, Resource}
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
@ -15,7 +16,11 @@ import docspell.store.{AddResult, Store}
|
|||||||
|
|
||||||
trait OEquipment[F[_]] {
|
trait OEquipment[F[_]] {
|
||||||
|
|
||||||
def findAll(account: AccountId, nameQuery: Option[String]): F[Vector[REquipment]]
|
def findAll(
|
||||||
|
account: AccountId,
|
||||||
|
nameQuery: Option[String],
|
||||||
|
order: OEquipment.EquipmentOrder
|
||||||
|
): F[Vector[REquipment]]
|
||||||
|
|
||||||
def find(account: AccountId, id: Ident): F[Option[REquipment]]
|
def find(account: AccountId, id: Ident): F[Option[REquipment]]
|
||||||
|
|
||||||
@ -27,11 +32,39 @@ trait OEquipment[F[_]] {
|
|||||||
}
|
}
|
||||||
|
|
||||||
object OEquipment {
|
object OEquipment {
|
||||||
|
import docspell.store.qb.DSL._
|
||||||
|
|
||||||
|
sealed trait EquipmentOrder
|
||||||
|
object EquipmentOrder {
|
||||||
|
final case object NameAsc extends EquipmentOrder
|
||||||
|
final case object NameDesc extends EquipmentOrder
|
||||||
|
|
||||||
|
def parse(str: String): Either[String, EquipmentOrder] =
|
||||||
|
str.toLowerCase match {
|
||||||
|
case "name" => Right(NameAsc)
|
||||||
|
case "-name" => Right(NameDesc)
|
||||||
|
case _ => Left(s"Unknown sort property for equipments: $str")
|
||||||
|
}
|
||||||
|
|
||||||
|
def parseOrDefault(str: String): EquipmentOrder =
|
||||||
|
parse(str).toOption.getOrElse(NameAsc)
|
||||||
|
|
||||||
|
private[ops] def apply(order: EquipmentOrder)(table: REquipment.Table) = order match {
|
||||||
|
case NameAsc => NonEmptyList.of(table.name.asc)
|
||||||
|
case NameDesc => NonEmptyList.of(table.name.desc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def apply[F[_]: Async](store: Store[F]): Resource[F, OEquipment[F]] =
|
def apply[F[_]: Async](store: Store[F]): Resource[F, OEquipment[F]] =
|
||||||
Resource.pure[F, OEquipment[F]](new OEquipment[F] {
|
Resource.pure[F, OEquipment[F]](new OEquipment[F] {
|
||||||
def findAll(account: AccountId, nameQuery: Option[String]): F[Vector[REquipment]] =
|
def findAll(
|
||||||
store.transact(REquipment.findAll(account.collective, nameQuery, _.name))
|
account: AccountId,
|
||||||
|
nameQuery: Option[String],
|
||||||
|
order: EquipmentOrder
|
||||||
|
): F[Vector[REquipment]] =
|
||||||
|
store.transact(
|
||||||
|
REquipment.findAll(account.collective, nameQuery, EquipmentOrder(order))
|
||||||
|
)
|
||||||
|
|
||||||
def find(account: AccountId, id: Ident): F[Option[REquipment]] =
|
def find(account: AccountId, id: Ident): F[Option[REquipment]] =
|
||||||
store.transact(REquipment.findById(id)).map(_.filter(_.cid == account.collective))
|
store.transact(REquipment.findById(id)).map(_.filter(_.cid == account.collective))
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
package docspell.backend.ops
|
package docspell.backend.ops
|
||||||
|
|
||||||
|
import cats.data.{NonEmptyList => Nel}
|
||||||
import cats.effect._
|
import cats.effect._
|
||||||
|
|
||||||
import docspell.common._
|
import docspell.common._
|
||||||
@ -18,7 +19,8 @@ trait OFolder[F[_]] {
|
|||||||
def findAll(
|
def findAll(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
ownerLogin: Option[Ident],
|
ownerLogin: Option[Ident],
|
||||||
nameQuery: Option[String]
|
query: Option[String],
|
||||||
|
order: OFolder.FolderOrder
|
||||||
): F[Vector[OFolder.FolderItem]]
|
): F[Vector[OFolder.FolderItem]]
|
||||||
|
|
||||||
def findById(id: Ident, account: AccountId): F[Option[OFolder.FolderDetail]]
|
def findById(id: Ident, account: AccountId): F[Option[OFolder.FolderDetail]]
|
||||||
@ -50,6 +52,7 @@ trait OFolder[F[_]] {
|
|||||||
}
|
}
|
||||||
|
|
||||||
object OFolder {
|
object OFolder {
|
||||||
|
import docspell.store.qb.DSL._
|
||||||
|
|
||||||
type FolderChangeResult = QFolder.FolderChangeResult
|
type FolderChangeResult = QFolder.FolderChangeResult
|
||||||
val FolderChangeResult = QFolder.FolderChangeResult
|
val FolderChangeResult = QFolder.FolderChangeResult
|
||||||
@ -60,14 +63,45 @@ object OFolder {
|
|||||||
type FolderDetail = QFolder.FolderDetail
|
type FolderDetail = QFolder.FolderDetail
|
||||||
val FolderDetail = QFolder.FolderDetail
|
val FolderDetail = QFolder.FolderDetail
|
||||||
|
|
||||||
|
sealed trait FolderOrder
|
||||||
|
object FolderOrder {
|
||||||
|
final case object NameAsc extends FolderOrder
|
||||||
|
final case object NameDesc extends FolderOrder
|
||||||
|
final case object OwnerAsc extends FolderOrder
|
||||||
|
final case object OwnerDesc extends FolderOrder
|
||||||
|
|
||||||
|
def parse(str: String): Either[String, FolderOrder] =
|
||||||
|
str.toLowerCase match {
|
||||||
|
case "name" => Right(NameAsc)
|
||||||
|
case "-name" => Right(NameDesc)
|
||||||
|
case "owner" => Right(OwnerAsc)
|
||||||
|
case "-owner" => Right(OwnerDesc)
|
||||||
|
case _ => Left(s"Unknown sort property for folder: $str")
|
||||||
|
}
|
||||||
|
|
||||||
|
def parseOrDefault(str: String): FolderOrder =
|
||||||
|
parse(str).toOption.getOrElse(NameAsc)
|
||||||
|
|
||||||
|
private[ops] def apply(order: FolderOrder)(folder: RFolder.Table, user: RUser.Table) =
|
||||||
|
order match {
|
||||||
|
case NameAsc => Nel.of(folder.name.asc)
|
||||||
|
case NameDesc => Nel.of(folder.name.desc)
|
||||||
|
case OwnerAsc => Nel.of(user.login.asc, folder.name.asc)
|
||||||
|
case OwnerDesc => Nel.of(user.login.desc, folder.name.desc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def apply[F[_]](store: Store[F]): Resource[F, OFolder[F]] =
|
def apply[F[_]](store: Store[F]): Resource[F, OFolder[F]] =
|
||||||
Resource.pure[F, OFolder[F]](new OFolder[F] {
|
Resource.pure[F, OFolder[F]](new OFolder[F] {
|
||||||
def findAll(
|
def findAll(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
ownerLogin: Option[Ident],
|
ownerLogin: Option[Ident],
|
||||||
nameQuery: Option[String]
|
query: Option[String],
|
||||||
|
order: FolderOrder
|
||||||
): F[Vector[FolderItem]] =
|
): F[Vector[FolderItem]] =
|
||||||
store.transact(QFolder.findAll(account, None, ownerLogin, nameQuery))
|
store.transact(
|
||||||
|
QFolder.findAll(account, None, ownerLogin, query, FolderOrder(order))
|
||||||
|
)
|
||||||
|
|
||||||
def findById(id: Ident, account: AccountId): F[Option[FolderDetail]] =
|
def findById(id: Ident, account: AccountId): F[Option[FolderDetail]] =
|
||||||
store.transact(QFolder.findById(id, account))
|
store.transact(QFolder.findById(id, account))
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
package docspell.backend.ops
|
package docspell.backend.ops
|
||||||
|
|
||||||
|
import cats.data.NonEmptyList
|
||||||
import cats.effect.{Async, Resource}
|
import cats.effect.{Async, Resource}
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
@ -16,10 +17,18 @@ import docspell.store.queries.QOrganization
|
|||||||
import docspell.store.records._
|
import docspell.store.records._
|
||||||
|
|
||||||
trait OOrganization[F[_]] {
|
trait OOrganization[F[_]] {
|
||||||
def findAllOrg(account: AccountId, query: Option[String]): F[Vector[OrgAndContacts]]
|
def findAllOrg(
|
||||||
|
account: AccountId,
|
||||||
|
query: Option[String],
|
||||||
|
order: OrganizationOrder
|
||||||
|
): F[Vector[OrgAndContacts]]
|
||||||
def findOrg(account: AccountId, orgId: Ident): F[Option[OrgAndContacts]]
|
def findOrg(account: AccountId, orgId: Ident): F[Option[OrgAndContacts]]
|
||||||
|
|
||||||
def findAllOrgRefs(account: AccountId, nameQuery: Option[String]): F[Vector[IdRef]]
|
def findAllOrgRefs(
|
||||||
|
account: AccountId,
|
||||||
|
nameQuery: Option[String],
|
||||||
|
order: OrganizationOrder
|
||||||
|
): F[Vector[IdRef]]
|
||||||
|
|
||||||
def addOrg(s: OrgAndContacts): F[AddResult]
|
def addOrg(s: OrgAndContacts): F[AddResult]
|
||||||
|
|
||||||
@ -27,12 +36,17 @@ trait OOrganization[F[_]] {
|
|||||||
|
|
||||||
def findAllPerson(
|
def findAllPerson(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
query: Option[String]
|
query: Option[String],
|
||||||
|
order: PersonOrder
|
||||||
): F[Vector[PersonAndContacts]]
|
): F[Vector[PersonAndContacts]]
|
||||||
|
|
||||||
def findPerson(account: AccountId, persId: Ident): F[Option[PersonAndContacts]]
|
def findPerson(account: AccountId, persId: Ident): F[Option[PersonAndContacts]]
|
||||||
|
|
||||||
def findAllPersonRefs(account: AccountId, nameQuery: Option[String]): F[Vector[IdRef]]
|
def findAllPersonRefs(
|
||||||
|
account: AccountId,
|
||||||
|
nameQuery: Option[String],
|
||||||
|
order: PersonOrder
|
||||||
|
): F[Vector[IdRef]]
|
||||||
|
|
||||||
/** Add a new person with their contacts. The additional organization is ignored. */
|
/** Add a new person with their contacts. The additional organization is ignored. */
|
||||||
def addPerson(s: PersonAndContacts): F[AddResult]
|
def addPerson(s: PersonAndContacts): F[AddResult]
|
||||||
@ -46,6 +60,7 @@ trait OOrganization[F[_]] {
|
|||||||
}
|
}
|
||||||
|
|
||||||
object OOrganization {
|
object OOrganization {
|
||||||
|
import docspell.store.qb.DSL._
|
||||||
|
|
||||||
case class OrgAndContacts(org: ROrganization, contacts: Seq[RContact])
|
case class OrgAndContacts(org: ROrganization, contacts: Seq[RContact])
|
||||||
|
|
||||||
@ -55,15 +70,79 @@ object OOrganization {
|
|||||||
contacts: Seq[RContact]
|
contacts: Seq[RContact]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
sealed trait OrganizationOrder
|
||||||
|
object OrganizationOrder {
|
||||||
|
final case object NameAsc extends OrganizationOrder
|
||||||
|
final case object NameDesc extends OrganizationOrder
|
||||||
|
|
||||||
|
def parse(str: String): Either[String, OrganizationOrder] =
|
||||||
|
str.toLowerCase match {
|
||||||
|
case "name" => Right(NameAsc)
|
||||||
|
case "-name" => Right(NameDesc)
|
||||||
|
case _ => Left(s"Unknown sort property for organization: $str")
|
||||||
|
}
|
||||||
|
|
||||||
|
def parseOrDefault(str: String): OrganizationOrder =
|
||||||
|
parse(str).toOption.getOrElse(NameAsc)
|
||||||
|
|
||||||
|
private[ops] def apply(order: OrganizationOrder)(table: ROrganization.Table) =
|
||||||
|
order match {
|
||||||
|
case NameAsc => NonEmptyList.of(table.name.asc)
|
||||||
|
case NameDesc => NonEmptyList.of(table.name.desc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed trait PersonOrder
|
||||||
|
object PersonOrder {
|
||||||
|
final case object NameAsc extends PersonOrder
|
||||||
|
final case object NameDesc extends PersonOrder
|
||||||
|
final case object OrgAsc extends PersonOrder
|
||||||
|
final case object OrgDesc extends PersonOrder
|
||||||
|
|
||||||
|
def parse(str: String): Either[String, PersonOrder] =
|
||||||
|
str.toLowerCase match {
|
||||||
|
case "name" => Right(NameAsc)
|
||||||
|
case "-name" => Right(NameDesc)
|
||||||
|
case "org" => Right(OrgAsc)
|
||||||
|
case "-org" => Right(OrgDesc)
|
||||||
|
case _ => Left(s"Unknown sort property for person: $str")
|
||||||
|
}
|
||||||
|
|
||||||
|
def parseOrDefault(str: String): PersonOrder =
|
||||||
|
parse(str).toOption.getOrElse(NameAsc)
|
||||||
|
|
||||||
|
private[ops] def apply(
|
||||||
|
order: PersonOrder
|
||||||
|
)(person: RPerson.Table, org: ROrganization.Table) =
|
||||||
|
order match {
|
||||||
|
case NameAsc => NonEmptyList.of(person.name.asc)
|
||||||
|
case NameDesc => NonEmptyList.of(person.name.desc)
|
||||||
|
case OrgAsc => NonEmptyList.of(org.name.asc)
|
||||||
|
case OrgDesc => NonEmptyList.of(org.name.desc)
|
||||||
|
}
|
||||||
|
|
||||||
|
private[ops] def nameOnly(order: PersonOrder)(person: RPerson.Table) =
|
||||||
|
order match {
|
||||||
|
case NameAsc => NonEmptyList.of(person.name.asc)
|
||||||
|
case NameDesc => NonEmptyList.of(person.name.desc)
|
||||||
|
case OrgAsc => NonEmptyList.of(person.name.asc)
|
||||||
|
case OrgDesc => NonEmptyList.of(person.name.asc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def apply[F[_]: Async](store: Store[F]): Resource[F, OOrganization[F]] =
|
def apply[F[_]: Async](store: Store[F]): Resource[F, OOrganization[F]] =
|
||||||
Resource.pure[F, OOrganization[F]](new OOrganization[F] {
|
Resource.pure[F, OOrganization[F]](new OOrganization[F] {
|
||||||
|
|
||||||
def findAllOrg(
|
def findAllOrg(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
query: Option[String]
|
query: Option[String],
|
||||||
|
order: OrganizationOrder
|
||||||
): F[Vector[OrgAndContacts]] =
|
): F[Vector[OrgAndContacts]] =
|
||||||
store
|
store
|
||||||
.transact(QOrganization.findOrgAndContact(account.collective, query, _.name))
|
.transact(
|
||||||
|
QOrganization
|
||||||
|
.findOrgAndContact(account.collective, query, OrganizationOrder(order))
|
||||||
|
)
|
||||||
.map { case (org, cont) => OrgAndContacts(org, cont) }
|
.map { case (org, cont) => OrgAndContacts(org, cont) }
|
||||||
.compile
|
.compile
|
||||||
.toVector
|
.toVector
|
||||||
@ -75,9 +154,16 @@ object OOrganization {
|
|||||||
|
|
||||||
def findAllOrgRefs(
|
def findAllOrgRefs(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
nameQuery: Option[String]
|
nameQuery: Option[String],
|
||||||
|
order: OrganizationOrder
|
||||||
): F[Vector[IdRef]] =
|
): F[Vector[IdRef]] =
|
||||||
store.transact(ROrganization.findAllRef(account.collective, nameQuery, _.name))
|
store.transact(
|
||||||
|
ROrganization.findAllRef(
|
||||||
|
account.collective,
|
||||||
|
nameQuery,
|
||||||
|
OrganizationOrder(order)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
def addOrg(s: OrgAndContacts): F[AddResult] =
|
def addOrg(s: OrgAndContacts): F[AddResult] =
|
||||||
QOrganization.addOrg(s.org, s.contacts, s.org.cid)(store)
|
QOrganization.addOrg(s.org, s.contacts, s.org.cid)(store)
|
||||||
@ -87,10 +173,14 @@ object OOrganization {
|
|||||||
|
|
||||||
def findAllPerson(
|
def findAllPerson(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
query: Option[String]
|
query: Option[String],
|
||||||
|
order: PersonOrder
|
||||||
): F[Vector[PersonAndContacts]] =
|
): F[Vector[PersonAndContacts]] =
|
||||||
store
|
store
|
||||||
.transact(QOrganization.findPersonAndContact(account.collective, query, _.name))
|
.transact(
|
||||||
|
QOrganization
|
||||||
|
.findPersonAndContact(account.collective, query, PersonOrder(order))
|
||||||
|
)
|
||||||
.map { case (person, org, cont) => PersonAndContacts(person, org, cont) }
|
.map { case (person, org, cont) => PersonAndContacts(person, org, cont) }
|
||||||
.compile
|
.compile
|
||||||
.toVector
|
.toVector
|
||||||
@ -102,9 +192,12 @@ object OOrganization {
|
|||||||
|
|
||||||
def findAllPersonRefs(
|
def findAllPersonRefs(
|
||||||
account: AccountId,
|
account: AccountId,
|
||||||
nameQuery: Option[String]
|
nameQuery: Option[String],
|
||||||
|
order: PersonOrder
|
||||||
): F[Vector[IdRef]] =
|
): F[Vector[IdRef]] =
|
||||||
store.transact(RPerson.findAllRef(account.collective, nameQuery, _.name))
|
store.transact(
|
||||||
|
RPerson.findAllRef(account.collective, nameQuery, PersonOrder.nameOnly(order))
|
||||||
|
)
|
||||||
|
|
||||||
def addPerson(s: PersonAndContacts): F[AddResult] =
|
def addPerson(s: PersonAndContacts): F[AddResult] =
|
||||||
QOrganization.addPerson(s.person, s.contacts, s.person.cid)(store)
|
QOrganization.addPerson(s.person, s.contacts, s.person.cid)(store)
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
package docspell.backend.ops
|
package docspell.backend.ops
|
||||||
|
|
||||||
|
import cats.data.NonEmptyList
|
||||||
import cats.effect.{Async, Resource}
|
import cats.effect.{Async, Resource}
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
@ -16,7 +17,11 @@ import docspell.store.{AddResult, Store}
|
|||||||
|
|
||||||
trait OTag[F[_]] {
|
trait OTag[F[_]] {
|
||||||
|
|
||||||
def findAll(account: AccountId, nameQuery: Option[String]): F[Vector[RTag]]
|
def findAll(
|
||||||
|
account: AccountId,
|
||||||
|
query: Option[String],
|
||||||
|
order: OTag.TagOrder
|
||||||
|
): F[Vector[RTag]]
|
||||||
|
|
||||||
def add(s: RTag): F[AddResult]
|
def add(s: RTag): F[AddResult]
|
||||||
|
|
||||||
@ -30,11 +35,43 @@ trait OTag[F[_]] {
|
|||||||
}
|
}
|
||||||
|
|
||||||
object OTag {
|
object OTag {
|
||||||
|
import docspell.store.qb.DSL._
|
||||||
|
|
||||||
|
sealed trait TagOrder
|
||||||
|
object TagOrder {
|
||||||
|
final case object NameAsc extends TagOrder
|
||||||
|
final case object NameDesc extends TagOrder
|
||||||
|
final case object CategoryAsc extends TagOrder
|
||||||
|
final case object CategoryDesc extends TagOrder
|
||||||
|
|
||||||
|
def parse(str: String): Either[String, TagOrder] =
|
||||||
|
str.toLowerCase match {
|
||||||
|
case "name" => Right(NameAsc)
|
||||||
|
case "-name" => Right(NameDesc)
|
||||||
|
case "category" => Right(CategoryAsc)
|
||||||
|
case "-category" => Right(CategoryDesc)
|
||||||
|
case _ => Left(s"Unknown sort property for tags: $str")
|
||||||
|
}
|
||||||
|
|
||||||
|
def parseOrDefault(str: String): TagOrder =
|
||||||
|
parse(str).toOption.getOrElse(NameAsc)
|
||||||
|
|
||||||
|
private[ops] def apply(order: TagOrder)(table: RTag.Table) = order match {
|
||||||
|
case NameAsc => NonEmptyList.of(table.name.asc)
|
||||||
|
case CategoryAsc => NonEmptyList.of(table.category.asc, table.name.asc)
|
||||||
|
case NameDesc => NonEmptyList.of(table.name.desc)
|
||||||
|
case CategoryDesc => NonEmptyList.of(table.category.desc, table.name.desc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def apply[F[_]: Async](store: Store[F]): Resource[F, OTag[F]] =
|
def apply[F[_]: Async](store: Store[F]): Resource[F, OTag[F]] =
|
||||||
Resource.pure[F, OTag[F]](new OTag[F] {
|
Resource.pure[F, OTag[F]](new OTag[F] {
|
||||||
def findAll(account: AccountId, nameQuery: Option[String]): F[Vector[RTag]] =
|
def findAll(
|
||||||
store.transact(RTag.findAll(account.collective, nameQuery, _.name))
|
account: AccountId,
|
||||||
|
query: Option[String],
|
||||||
|
order: TagOrder
|
||||||
|
): F[Vector[RTag]] =
|
||||||
|
store.transact(RTag.findAll(account.collective, query, TagOrder(order)))
|
||||||
|
|
||||||
def add(t: RTag): F[AddResult] = {
|
def add(t: RTag): F[AddResult] = {
|
||||||
def insert = RTag.insert(t)
|
def insert = RTag.insert(t)
|
||||||
|
@ -501,11 +501,14 @@ paths:
|
|||||||
tags: [ Tags ]
|
tags: [ Tags ]
|
||||||
summary: Get a list of tags
|
summary: Get a list of tags
|
||||||
description: |
|
description: |
|
||||||
Return a list of all configured tags.
|
Return a list of all configured tags. The `sort` query
|
||||||
|
parameter is optional and can specify how the list is sorted.
|
||||||
|
Possible values are: `name`, `-name`, `category`, `-category`.
|
||||||
security:
|
security:
|
||||||
- authTokenHeader: []
|
- authTokenHeader: []
|
||||||
parameters:
|
parameters:
|
||||||
- $ref: "#/components/parameters/q"
|
- $ref: "#/components/parameters/q"
|
||||||
|
- $ref: "#/components/parameters/sort"
|
||||||
responses:
|
responses:
|
||||||
200:
|
200:
|
||||||
description: Ok
|
description: Ok
|
||||||
@ -579,12 +582,16 @@ paths:
|
|||||||
tags: [ Organization ]
|
tags: [ Organization ]
|
||||||
summary: Get a list of organizations.
|
summary: Get a list of organizations.
|
||||||
description: |
|
description: |
|
||||||
Return a list of all organizations. Only name and id are returned.
|
Return a list of all organizations. Only name and id are
|
||||||
|
returned. If `full` is specified, the list contains all
|
||||||
|
organization data. The `sort` parameter can be either `name`
|
||||||
|
or `-name` to specify the order.
|
||||||
security:
|
security:
|
||||||
- authTokenHeader: []
|
- authTokenHeader: []
|
||||||
parameters:
|
parameters:
|
||||||
- $ref: "#/components/parameters/full"
|
- $ref: "#/components/parameters/full"
|
||||||
- $ref: "#/components/parameters/q"
|
- $ref: "#/components/parameters/q"
|
||||||
|
- $ref: "#/components/parameters/sort"
|
||||||
responses:
|
responses:
|
||||||
200:
|
200:
|
||||||
description: Ok
|
description: Ok
|
||||||
@ -677,12 +684,17 @@ paths:
|
|||||||
tags: [ Person ]
|
tags: [ Person ]
|
||||||
summary: Get a list of persons.
|
summary: Get a list of persons.
|
||||||
description: |
|
description: |
|
||||||
Return a list of all persons. Only name and id are returned.
|
Return a list of all persons. Only name and id are returned
|
||||||
|
unless the `full` parameter is specified. The `sort` parameter
|
||||||
|
can be used to control the order of the result. Use one of:
|
||||||
|
`name`, `-name`, `org`, `-org`. Note that order by `org` only
|
||||||
|
works when retrieving the full list.
|
||||||
security:
|
security:
|
||||||
- authTokenHeader: []
|
- authTokenHeader: []
|
||||||
parameters:
|
parameters:
|
||||||
- $ref: "#/components/parameters/full"
|
- $ref: "#/components/parameters/full"
|
||||||
- $ref: "#/components/parameters/q"
|
- $ref: "#/components/parameters/q"
|
||||||
|
- $ref: "#/components/parameters/sort"
|
||||||
responses:
|
responses:
|
||||||
200:
|
200:
|
||||||
description: Ok
|
description: Ok
|
||||||
@ -775,11 +787,14 @@ paths:
|
|||||||
tags: [ Equipment ]
|
tags: [ Equipment ]
|
||||||
summary: Get a list of equipments
|
summary: Get a list of equipments
|
||||||
description: |
|
description: |
|
||||||
Return a list of all configured equipments.
|
Return a list of all configured equipments. The sort query
|
||||||
|
parameter is optional and can be one of `name` or `-name` to
|
||||||
|
sort the list of equipments.
|
||||||
security:
|
security:
|
||||||
- authTokenHeader: []
|
- authTokenHeader: []
|
||||||
parameters:
|
parameters:
|
||||||
- $ref: "#/components/parameters/q"
|
- $ref: "#/components/parameters/q"
|
||||||
|
- $ref: "#/components/parameters/sort"
|
||||||
responses:
|
responses:
|
||||||
200:
|
200:
|
||||||
description: Ok
|
description: Ok
|
||||||
@ -3771,11 +3786,15 @@ paths:
|
|||||||
tags: [ Custom Fields ]
|
tags: [ Custom Fields ]
|
||||||
summary: Get all defined custom fields.
|
summary: Get all defined custom fields.
|
||||||
description: |
|
description: |
|
||||||
Get all custom fields defined for the current collective.
|
Get all custom fields defined for the current collective. The
|
||||||
|
`sort` parameter can be used to control the order of the
|
||||||
|
returned list. It can take a value from: `name`, `-name`,
|
||||||
|
`label`, `-label`, `type`, `-type`.
|
||||||
security:
|
security:
|
||||||
- authTokenHeader: []
|
- authTokenHeader: []
|
||||||
parameters:
|
parameters:
|
||||||
- $ref: "#/components/parameters/q"
|
- $ref: "#/components/parameters/q"
|
||||||
|
- $ref: "#/components/parameters/sort"
|
||||||
responses:
|
responses:
|
||||||
200:
|
200:
|
||||||
description: Ok
|
description: Ok
|
||||||
@ -5985,6 +6004,14 @@ components:
|
|||||||
schema:
|
schema:
|
||||||
type: integer
|
type: integer
|
||||||
format: int32
|
format: int32
|
||||||
|
sort:
|
||||||
|
name: sort
|
||||||
|
in: query
|
||||||
|
required: false
|
||||||
|
description: |
|
||||||
|
How to sort the returned list
|
||||||
|
schema:
|
||||||
|
type: string
|
||||||
withDetails:
|
withDetails:
|
||||||
name: withDetails
|
name: withDetails
|
||||||
in: query
|
in: query
|
||||||
|
@ -6,6 +6,11 @@
|
|||||||
|
|
||||||
package docspell.restserver.http4s
|
package docspell.restserver.http4s
|
||||||
|
|
||||||
|
import docspell.backend.ops.OCustomFields.CustomFieldOrder
|
||||||
|
import docspell.backend.ops.OEquipment.EquipmentOrder
|
||||||
|
import docspell.backend.ops.OFolder.FolderOrder
|
||||||
|
import docspell.backend.ops.OOrganization.{OrganizationOrder, PersonOrder}
|
||||||
|
import docspell.backend.ops.OTag.TagOrder
|
||||||
import docspell.common.ContactKind
|
import docspell.common.ContactKind
|
||||||
import docspell.common.SearchMode
|
import docspell.common.SearchMode
|
||||||
|
|
||||||
@ -29,6 +34,36 @@ object QueryParam {
|
|||||||
SearchMode.fromString(str).left.map(s => ParseFailure(str, s))
|
SearchMode.fromString(str).left.map(s => ParseFailure(str, s))
|
||||||
)
|
)
|
||||||
|
|
||||||
|
implicit val tagOrderDecoder: QueryParamDecoder[TagOrder] =
|
||||||
|
QueryParamDecoder[String].emap(str =>
|
||||||
|
TagOrder.parse(str).left.map(s => ParseFailure(str, s))
|
||||||
|
)
|
||||||
|
|
||||||
|
implicit val euqipOrderDecoder: QueryParamDecoder[EquipmentOrder] =
|
||||||
|
QueryParamDecoder[String].emap(str =>
|
||||||
|
EquipmentOrder.parse(str).left.map(s => ParseFailure(str, s))
|
||||||
|
)
|
||||||
|
|
||||||
|
implicit val orgOrderDecoder: QueryParamDecoder[OrganizationOrder] =
|
||||||
|
QueryParamDecoder[String].emap(str =>
|
||||||
|
OrganizationOrder.parse(str).left.map(s => ParseFailure(str, s))
|
||||||
|
)
|
||||||
|
|
||||||
|
implicit val personOrderDecoder: QueryParamDecoder[PersonOrder] =
|
||||||
|
QueryParamDecoder[String].emap(str =>
|
||||||
|
PersonOrder.parse(str).left.map(s => ParseFailure(str, s))
|
||||||
|
)
|
||||||
|
|
||||||
|
implicit val folderOrderDecoder: QueryParamDecoder[FolderOrder] =
|
||||||
|
QueryParamDecoder[String].emap(str =>
|
||||||
|
FolderOrder.parse(str).left.map(s => ParseFailure(str, s))
|
||||||
|
)
|
||||||
|
|
||||||
|
implicit val customFieldOrderDecoder: QueryParamDecoder[CustomFieldOrder] =
|
||||||
|
QueryParamDecoder[String].emap(str =>
|
||||||
|
CustomFieldOrder.parse(str).left.map(s => ParseFailure(str, s))
|
||||||
|
)
|
||||||
|
|
||||||
object FullOpt extends OptionalQueryParamDecoderMatcher[Boolean]("full")
|
object FullOpt extends OptionalQueryParamDecoderMatcher[Boolean]("full")
|
||||||
|
|
||||||
object OwningOpt extends OptionalQueryParamDecoderMatcher[Boolean]("owning")
|
object OwningOpt extends OptionalQueryParamDecoderMatcher[Boolean]("owning")
|
||||||
@ -42,6 +77,12 @@ object QueryParam {
|
|||||||
object Offset extends OptionalQueryParamDecoderMatcher[Int]("offset")
|
object Offset extends OptionalQueryParamDecoderMatcher[Int]("offset")
|
||||||
object WithDetails extends OptionalQueryParamDecoderMatcher[Boolean]("withDetails")
|
object WithDetails extends OptionalQueryParamDecoderMatcher[Boolean]("withDetails")
|
||||||
object SearchKind extends OptionalQueryParamDecoderMatcher[SearchMode]("searchMode")
|
object SearchKind extends OptionalQueryParamDecoderMatcher[SearchMode]("searchMode")
|
||||||
|
object TagSort extends OptionalQueryParamDecoderMatcher[TagOrder]("sort")
|
||||||
|
object EquipSort extends OptionalQueryParamDecoderMatcher[EquipmentOrder]("sort")
|
||||||
|
object OrgSort extends OptionalQueryParamDecoderMatcher[OrganizationOrder]("sort")
|
||||||
|
object PersonSort extends OptionalQueryParamDecoderMatcher[PersonOrder]("sort")
|
||||||
|
object FolderSort extends OptionalQueryParamDecoderMatcher[FolderOrder]("sort")
|
||||||
|
object FieldSort extends OptionalQueryParamDecoderMatcher[CustomFieldOrder]("sort")
|
||||||
|
|
||||||
object WithFallback extends OptionalQueryParamDecoderMatcher[Boolean]("withFallback")
|
object WithFallback extends OptionalQueryParamDecoderMatcher[Boolean]("withFallback")
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,7 @@ import cats.implicits._
|
|||||||
import docspell.backend.BackendApp
|
import docspell.backend.BackendApp
|
||||||
import docspell.backend.auth.AuthToken
|
import docspell.backend.auth.AuthToken
|
||||||
import docspell.backend.ops.OCustomFields
|
import docspell.backend.ops.OCustomFields
|
||||||
import docspell.backend.ops.OCustomFields.CustomFieldData
|
import docspell.backend.ops.OCustomFields.{CustomFieldData, CustomFieldOrder}
|
||||||
import docspell.common._
|
import docspell.common._
|
||||||
import docspell.restapi.model._
|
import docspell.restapi.model._
|
||||||
import docspell.restserver.conv.Conversions
|
import docspell.restserver.conv.Conversions
|
||||||
@ -34,9 +34,14 @@ object CustomFieldRoutes {
|
|||||||
import dsl._
|
import dsl._
|
||||||
|
|
||||||
HttpRoutes.of {
|
HttpRoutes.of {
|
||||||
case GET -> Root :? QueryParam.QueryOpt(param) =>
|
case GET -> Root :? QueryParam.QueryOpt(param) +& QueryParam.FieldSort(sort) =>
|
||||||
|
val order = sort.getOrElse(CustomFieldOrder.NameAsc)
|
||||||
for {
|
for {
|
||||||
fs <- backend.customFields.findAll(user.account.collective, param.map(_.q))
|
fs <- backend.customFields.findAll(
|
||||||
|
user.account.collective,
|
||||||
|
param.map(_.q),
|
||||||
|
order
|
||||||
|
)
|
||||||
res <- Ok(CustomFieldList(fs.map(convertField).toList))
|
res <- Ok(CustomFieldList(fs.map(convertField).toList))
|
||||||
} yield res
|
} yield res
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@ import cats.implicits._
|
|||||||
|
|
||||||
import docspell.backend.BackendApp
|
import docspell.backend.BackendApp
|
||||||
import docspell.backend.auth.AuthToken
|
import docspell.backend.auth.AuthToken
|
||||||
|
import docspell.backend.ops.OEquipment
|
||||||
import docspell.common.Ident
|
import docspell.common.Ident
|
||||||
import docspell.restapi.model._
|
import docspell.restapi.model._
|
||||||
import docspell.restserver.conv.Conversions._
|
import docspell.restserver.conv.Conversions._
|
||||||
@ -29,9 +30,13 @@ object EquipmentRoutes {
|
|||||||
import dsl._
|
import dsl._
|
||||||
|
|
||||||
HttpRoutes.of {
|
HttpRoutes.of {
|
||||||
case GET -> Root :? QueryParam.QueryOpt(q) =>
|
case GET -> Root :? QueryParam.QueryOpt(q) :? QueryParam.EquipSort(sort) =>
|
||||||
for {
|
for {
|
||||||
data <- backend.equipment.findAll(user.account, q.map(_.q))
|
data <- backend.equipment.findAll(
|
||||||
|
user.account,
|
||||||
|
q.map(_.q),
|
||||||
|
sort.getOrElse(OEquipment.EquipmentOrder.NameAsc)
|
||||||
|
)
|
||||||
resp <- Ok(EquipmentList(data.map(mkEquipment).toList))
|
resp <- Ok(EquipmentList(data.map(mkEquipment).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
|
|
||||||
|
@ -31,11 +31,13 @@ object FolderRoutes {
|
|||||||
import dsl._
|
import dsl._
|
||||||
|
|
||||||
HttpRoutes.of {
|
HttpRoutes.of {
|
||||||
case GET -> Root :? QueryParam.QueryOpt(q) :? QueryParam.OwningOpt(owning) =>
|
case GET -> Root :? QueryParam.QueryOpt(q) :?
|
||||||
|
QueryParam.OwningOpt(owning) +& QueryParam.FolderSort(sort) =>
|
||||||
|
val order = sort.getOrElse(OFolder.FolderOrder.NameAsc)
|
||||||
val login =
|
val login =
|
||||||
owning.filter(identity).map(_ => user.account.user)
|
owning.filter(identity).map(_ => user.account.user)
|
||||||
for {
|
for {
|
||||||
all <- backend.folder.findAll(user.account, login, q.map(_.q))
|
all <- backend.folder.findAll(user.account, login, q.map(_.q), order)
|
||||||
resp <- Ok(FolderList(all.map(mkFolder).toList))
|
resp <- Ok(FolderList(all.map(mkFolder).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@ import cats.implicits._
|
|||||||
|
|
||||||
import docspell.backend.BackendApp
|
import docspell.backend.BackendApp
|
||||||
import docspell.backend.auth.AuthToken
|
import docspell.backend.auth.AuthToken
|
||||||
|
import docspell.backend.ops.OOrganization.OrganizationOrder
|
||||||
import docspell.common.Ident
|
import docspell.common.Ident
|
||||||
import docspell.restapi.model._
|
import docspell.restapi.model._
|
||||||
import docspell.restserver.conv.Conversions._
|
import docspell.restserver.conv.Conversions._
|
||||||
@ -29,15 +30,21 @@ object OrganizationRoutes {
|
|||||||
import dsl._
|
import dsl._
|
||||||
|
|
||||||
HttpRoutes.of {
|
HttpRoutes.of {
|
||||||
case GET -> Root :? QueryParam.FullOpt(full) +& QueryParam.QueryOpt(q) =>
|
case GET -> Root :? QueryParam.FullOpt(full) +&
|
||||||
|
QueryParam.QueryOpt(q) +& QueryParam.OrgSort(sort) =>
|
||||||
|
val order = sort.getOrElse(OrganizationOrder.NameAsc)
|
||||||
if (full.getOrElse(false))
|
if (full.getOrElse(false))
|
||||||
for {
|
for {
|
||||||
data <- backend.organization.findAllOrg(user.account, q.map(_.q))
|
data <- backend.organization.findAllOrg(
|
||||||
|
user.account,
|
||||||
|
q.map(_.q),
|
||||||
|
order
|
||||||
|
)
|
||||||
resp <- Ok(OrganizationList(data.map(mkOrg).toList))
|
resp <- Ok(OrganizationList(data.map(mkOrg).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
else
|
else
|
||||||
for {
|
for {
|
||||||
data <- backend.organization.findAllOrgRefs(user.account, q.map(_.q))
|
data <- backend.organization.findAllOrgRefs(user.account, q.map(_.q), order)
|
||||||
resp <- Ok(ReferenceList(data.map(mkIdName).toList))
|
resp <- Ok(ReferenceList(data.map(mkIdName).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@ import cats.implicits._
|
|||||||
|
|
||||||
import docspell.backend.BackendApp
|
import docspell.backend.BackendApp
|
||||||
import docspell.backend.auth.AuthToken
|
import docspell.backend.auth.AuthToken
|
||||||
|
import docspell.backend.ops.OOrganization
|
||||||
import docspell.common.Ident
|
import docspell.common.Ident
|
||||||
import docspell.common.syntax.all._
|
import docspell.common.syntax.all._
|
||||||
import docspell.restapi.model._
|
import docspell.restapi.model._
|
||||||
@ -32,15 +33,25 @@ object PersonRoutes {
|
|||||||
import dsl._
|
import dsl._
|
||||||
|
|
||||||
HttpRoutes.of {
|
HttpRoutes.of {
|
||||||
case GET -> Root :? QueryParam.FullOpt(full) +& QueryParam.QueryOpt(q) =>
|
case GET -> Root :? QueryParam.FullOpt(full) +&
|
||||||
|
QueryParam.QueryOpt(q) +& QueryParam.PersonSort(sort) =>
|
||||||
|
val order = sort.getOrElse(OOrganization.PersonOrder.NameAsc)
|
||||||
if (full.getOrElse(false))
|
if (full.getOrElse(false))
|
||||||
for {
|
for {
|
||||||
data <- backend.organization.findAllPerson(user.account, q.map(_.q))
|
data <- backend.organization.findAllPerson(
|
||||||
|
user.account,
|
||||||
|
q.map(_.q),
|
||||||
|
order
|
||||||
|
)
|
||||||
resp <- Ok(PersonList(data.map(mkPerson).toList))
|
resp <- Ok(PersonList(data.map(mkPerson).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
else
|
else
|
||||||
for {
|
for {
|
||||||
data <- backend.organization.findAllPersonRefs(user.account, q.map(_.q))
|
data <- backend.organization.findAllPersonRefs(
|
||||||
|
user.account,
|
||||||
|
q.map(_.q),
|
||||||
|
order
|
||||||
|
)
|
||||||
resp <- Ok(ReferenceList(data.map(mkIdName).toList))
|
resp <- Ok(ReferenceList(data.map(mkIdName).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
|
|
||||||
|
@ -11,6 +11,7 @@ import cats.implicits._
|
|||||||
|
|
||||||
import docspell.backend.BackendApp
|
import docspell.backend.BackendApp
|
||||||
import docspell.backend.auth.AuthToken
|
import docspell.backend.auth.AuthToken
|
||||||
|
import docspell.backend.ops.OTag.TagOrder
|
||||||
import docspell.common.Ident
|
import docspell.common.Ident
|
||||||
import docspell.restapi.model._
|
import docspell.restapi.model._
|
||||||
import docspell.restserver.conv.Conversions._
|
import docspell.restserver.conv.Conversions._
|
||||||
@ -28,9 +29,13 @@ object TagRoutes {
|
|||||||
import dsl._
|
import dsl._
|
||||||
|
|
||||||
HttpRoutes.of {
|
HttpRoutes.of {
|
||||||
case GET -> Root :? QueryParam.QueryOpt(q) =>
|
case GET -> Root :? QueryParam.QueryOpt(q) :? QueryParam.TagSort(sort) =>
|
||||||
for {
|
for {
|
||||||
all <- backend.tag.findAll(user.account, q.map(_.q))
|
all <- backend.tag.findAll(
|
||||||
|
user.account,
|
||||||
|
q.map(_.q),
|
||||||
|
sort.getOrElse(TagOrder.NameAsc)
|
||||||
|
)
|
||||||
resp <- Ok(TagList(all.size, all.map(mkTag).toList))
|
resp <- Ok(TagList(all.size, all.map(mkTag).toList))
|
||||||
} yield resp
|
} yield resp
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@ case class Column[A](name: String, table: TableDef) {
|
|||||||
|
|
||||||
def cast[B]: Column[B] =
|
def cast[B]: Column[B] =
|
||||||
this.asInstanceOf[Column[B]]
|
this.asInstanceOf[Column[B]]
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
object Column {}
|
object Column {}
|
||||||
|
@ -303,6 +303,12 @@ trait DSL extends DoobieMeta {
|
|||||||
def as(otherCol: Column[_]): SelectExpr =
|
def as(otherCol: Column[_]): SelectExpr =
|
||||||
SelectExpr.SelectFun(dbf, Some(otherCol.name))
|
SelectExpr.SelectFun(dbf, Some(otherCol.name))
|
||||||
|
|
||||||
|
def asc: OrderBy =
|
||||||
|
OrderBy(SelectExpr.SelectFun(dbf, None), OrderBy.OrderType.Asc)
|
||||||
|
|
||||||
|
def desc: OrderBy =
|
||||||
|
OrderBy(SelectExpr.SelectFun(dbf, None), OrderBy.OrderType.Desc)
|
||||||
|
|
||||||
def ===[A](value: A)(implicit P: Put[A]): Condition =
|
def ===[A](value: A)(implicit P: Put[A]): Condition =
|
||||||
Condition.CompareFVal(dbf.s, Operator.Eq, value)
|
Condition.CompareFVal(dbf.s, Operator.Eq, value)
|
||||||
|
|
||||||
|
@ -135,6 +135,9 @@ object Select {
|
|||||||
|
|
||||||
def orderBy(ob: OrderBy, obs: OrderBy*): Ordered =
|
def orderBy(ob: OrderBy, obs: OrderBy*): Ordered =
|
||||||
Ordered(this, ob, obs.toVector)
|
Ordered(this, ob, obs.toVector)
|
||||||
|
|
||||||
|
def orderBy(ob: Nel[OrderBy]): Ordered =
|
||||||
|
Ordered(this, ob.head, ob.tail.toVector)
|
||||||
}
|
}
|
||||||
|
|
||||||
case class RawSelect(fragment: Fragment) extends Select {
|
case class RawSelect(fragment: Fragment) extends Select {
|
||||||
|
@ -24,9 +24,10 @@ object QCustomField {
|
|||||||
|
|
||||||
def findAllLike(
|
def findAllLike(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
nameQuery: Option[String]
|
nameQuery: Option[String],
|
||||||
|
order: RCustomField.Table => Nel[OrderBy]
|
||||||
): ConnectionIO[Vector[CustomFieldData]] =
|
): ConnectionIO[Vector[CustomFieldData]] =
|
||||||
findFragment(coll, nameQuery, None).build.query[CustomFieldData].to[Vector]
|
findFragment(coll, nameQuery, None, order).build.query[CustomFieldData].to[Vector]
|
||||||
|
|
||||||
def findById(field: Ident, collective: Ident): ConnectionIO[Option[CustomFieldData]] =
|
def findById(field: Ident, collective: Ident): ConnectionIO[Option[CustomFieldData]] =
|
||||||
findFragment(collective, None, field.some).build.query[CustomFieldData].option
|
findFragment(collective, None, field.some).build.query[CustomFieldData].option
|
||||||
@ -34,7 +35,8 @@ object QCustomField {
|
|||||||
private def findFragment(
|
private def findFragment(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
nameQuery: Option[String],
|
nameQuery: Option[String],
|
||||||
fieldId: Option[Ident]
|
fieldId: Option[Ident],
|
||||||
|
order: RCustomField.Table => Nel[OrderBy] = t => Nel.of(t.name.asc)
|
||||||
): Select = {
|
): Select = {
|
||||||
val nameFilter = nameQuery.map { q =>
|
val nameFilter = nameQuery.map { q =>
|
||||||
f.name.likes(q) || (f.label.isNotNull && f.label.like(q))
|
f.name.likes(q) || (f.label.isNotNull && f.label.like(q))
|
||||||
@ -46,7 +48,7 @@ object QCustomField {
|
|||||||
.leftJoin(v, f.id === v.field),
|
.leftJoin(v, f.id === v.field),
|
||||||
f.cid === coll &&? nameFilter &&? fieldId.map(fid => f.id === fid),
|
f.cid === coll &&? nameFilter &&? fieldId.map(fid => f.id === fid),
|
||||||
GroupBy(f.all)
|
GroupBy(f.all)
|
||||||
)
|
).orderBy(order(f))
|
||||||
}
|
}
|
||||||
|
|
||||||
final case class FieldValue(
|
final case class FieldValue(
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
package docspell.store.queries
|
package docspell.store.queries
|
||||||
|
|
||||||
import cats.data.OptionT
|
import cats.data.OptionT
|
||||||
|
import cats.data.{NonEmptyList => Nel}
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
import docspell.common._
|
import docspell.common._
|
||||||
@ -156,7 +157,10 @@ object QFolder {
|
|||||||
).query[IdRef].to[Vector]
|
).query[IdRef].to[Vector]
|
||||||
|
|
||||||
(for {
|
(for {
|
||||||
folder <- OptionT(findAll(account, Some(id), None, None).map(_.headOption))
|
folder <- OptionT(
|
||||||
|
findAll(account, Some(id), None, None, (ft, _) => Nel.of(ft.name.asc))
|
||||||
|
.map(_.headOption)
|
||||||
|
)
|
||||||
memb <- OptionT.liftF(memberQ)
|
memb <- OptionT.liftF(memberQ)
|
||||||
} yield folder.withMembers(memb.toList)).value
|
} yield folder.withMembers(memb.toList)).value
|
||||||
}
|
}
|
||||||
@ -165,7 +169,8 @@ object QFolder {
|
|||||||
account: AccountId,
|
account: AccountId,
|
||||||
idQ: Option[Ident],
|
idQ: Option[Ident],
|
||||||
ownerLogin: Option[Ident],
|
ownerLogin: Option[Ident],
|
||||||
nameQ: Option[String]
|
nameQ: Option[String],
|
||||||
|
order: (RFolder.Table, RUser.Table) => Nel[OrderBy]
|
||||||
): ConnectionIO[Vector[FolderItem]] = {
|
): ConnectionIO[Vector[FolderItem]] = {
|
||||||
// with memberlogin as
|
// with memberlogin as
|
||||||
// (select m.folder_id,u.login
|
// (select m.folder_id,u.login
|
||||||
@ -239,7 +244,7 @@ object QFolder {
|
|||||||
nameQ.map(q => folder.name.like(s"%${q.toLowerCase}%")) &&?
|
nameQ.map(q => folder.name.like(s"%${q.toLowerCase}%")) &&?
|
||||||
ownerLogin.map(login => user.login === login)
|
ownerLogin.map(login => user.login === login)
|
||||||
)
|
)
|
||||||
).orderBy(folder.name.asc)
|
).orderBy(order(folder, user))
|
||||||
).build.query[FolderItem].to[Vector]
|
).build.query[FolderItem].to[Vector]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
package docspell.store.queries
|
package docspell.store.queries
|
||||||
|
|
||||||
import cats.data.NonEmptyList
|
import cats.data.{NonEmptyList => Nel}
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
import fs2._
|
import fs2._
|
||||||
|
|
||||||
@ -27,7 +27,7 @@ object QOrganization {
|
|||||||
def findOrgAndContact(
|
def findOrgAndContact(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
query: Option[String],
|
query: Option[String],
|
||||||
order: ROrganization.Table => Column[_]
|
order: ROrganization.Table => Nel[OrderBy]
|
||||||
): Stream[ConnectionIO, (ROrganization, Vector[RContact])] = {
|
): Stream[ConnectionIO, (ROrganization, Vector[RContact])] = {
|
||||||
val valFilter = query.map { q =>
|
val valFilter = query.map { q =>
|
||||||
val v = s"%$q%"
|
val v = s"%$q%"
|
||||||
@ -74,18 +74,18 @@ object QOrganization {
|
|||||||
def findPersonAndContact(
|
def findPersonAndContact(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
query: Option[String],
|
query: Option[String],
|
||||||
order: RPerson.Table => Column[_]
|
order: (RPerson.Table, ROrganization.Table) => Nel[OrderBy]
|
||||||
): Stream[ConnectionIO, (RPerson, Option[ROrganization], Vector[RContact])] = {
|
): Stream[ConnectionIO, (RPerson, Option[ROrganization], Vector[RContact])] = {
|
||||||
val valFilter = query
|
val valFilter = query
|
||||||
.map(s => s"%$s%")
|
.map(s => s"%$s%")
|
||||||
.map(v => c.value.like(v) || p.name.like(v) || p.notes.like(v))
|
.map(v => c.value.like(v) || p.name.like(v) || org.name.like(v) || p.notes.like(v))
|
||||||
val sql = Select(
|
val sql = Select(
|
||||||
select(p.all, org.all, c.all),
|
select(p.all, org.all, c.all),
|
||||||
from(p)
|
from(p)
|
||||||
.leftJoin(org, org.oid === p.oid)
|
.leftJoin(org, org.oid === p.oid)
|
||||||
.leftJoin(c, c.personId === p.pid),
|
.leftJoin(c, c.personId === p.pid),
|
||||||
p.cid === coll &&? valFilter
|
p.cid === coll &&? valFilter
|
||||||
).orderBy(order(p))
|
).orderBy(order(p, org))
|
||||||
|
|
||||||
sql.build
|
sql.build
|
||||||
.query[(RPerson, Option[ROrganization], Option[RContact])]
|
.query[(RPerson, Option[ROrganization], Option[RContact])]
|
||||||
@ -128,7 +128,7 @@ object QOrganization {
|
|||||||
coll: Ident,
|
coll: Ident,
|
||||||
value: String,
|
value: String,
|
||||||
ck: Option[ContactKind],
|
ck: Option[ContactKind],
|
||||||
use: Option[NonEmptyList[PersonUse]]
|
use: Option[Nel[PersonUse]]
|
||||||
): Stream[ConnectionIO, RPerson] =
|
): Stream[ConnectionIO, RPerson] =
|
||||||
runDistinct(
|
runDistinct(
|
||||||
select(p.all),
|
select(p.all),
|
||||||
|
@ -87,7 +87,7 @@ object REquipment {
|
|||||||
def findAll(
|
def findAll(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
nameQ: Option[String],
|
nameQ: Option[String],
|
||||||
order: Table => Column[_]
|
order: Table => NonEmptyList[OrderBy]
|
||||||
): ConnectionIO[Vector[REquipment]] = {
|
): ConnectionIO[Vector[REquipment]] = {
|
||||||
val t = Table(None)
|
val t = Table(None)
|
||||||
|
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
package docspell.store.records
|
package docspell.store.records
|
||||||
|
|
||||||
import cats.Eq
|
import cats.Eq
|
||||||
import cats.data.NonEmptyList
|
import cats.data.{NonEmptyList => Nel}
|
||||||
import fs2.Stream
|
import fs2.Stream
|
||||||
|
|
||||||
import docspell.common.{IdRef, _}
|
import docspell.common.{IdRef, _}
|
||||||
@ -52,7 +52,7 @@ object ROrganization {
|
|||||||
val shortName = Column[String]("short_name", this)
|
val shortName = Column[String]("short_name", this)
|
||||||
val use = Column[OrgUse]("org_use", this)
|
val use = Column[OrgUse]("org_use", this)
|
||||||
val all =
|
val all =
|
||||||
NonEmptyList.of[Column[_]](
|
Nel.of[Column[_]](
|
||||||
oid,
|
oid,
|
||||||
cid,
|
cid,
|
||||||
name,
|
name,
|
||||||
@ -122,7 +122,7 @@ object ROrganization {
|
|||||||
def findLike(
|
def findLike(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
orgName: String,
|
orgName: String,
|
||||||
use: NonEmptyList[OrgUse]
|
use: Nel[OrgUse]
|
||||||
): ConnectionIO[Vector[IdRef]] =
|
): ConnectionIO[Vector[IdRef]] =
|
||||||
run(
|
run(
|
||||||
select(T.oid, T.name),
|
select(T.oid, T.name),
|
||||||
@ -163,7 +163,7 @@ object ROrganization {
|
|||||||
def findAllRef(
|
def findAllRef(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
nameQ: Option[String],
|
nameQ: Option[String],
|
||||||
order: Table => Column[_]
|
order: Table => Nel[OrderBy]
|
||||||
): ConnectionIO[Vector[IdRef]] = {
|
): ConnectionIO[Vector[IdRef]] = {
|
||||||
val nameFilter = nameQ.map(s =>
|
val nameFilter = nameQ.map(s =>
|
||||||
T.name.like(s"%${s.toLowerCase}%") || T.shortName.like(s"%${s.toLowerCase}%")
|
T.name.like(s"%${s.toLowerCase}%") || T.shortName.like(s"%${s.toLowerCase}%")
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
package docspell.store.records
|
package docspell.store.records
|
||||||
|
|
||||||
import cats.Eq
|
import cats.Eq
|
||||||
import cats.data.NonEmptyList
|
import cats.data.{NonEmptyList => Nel}
|
||||||
import cats.effect._
|
import cats.effect._
|
||||||
import fs2.Stream
|
import fs2.Stream
|
||||||
|
|
||||||
@ -52,7 +52,7 @@ object RPerson {
|
|||||||
val updated = Column[Timestamp]("updated", this)
|
val updated = Column[Timestamp]("updated", this)
|
||||||
val oid = Column[Ident]("oid", this)
|
val oid = Column[Ident]("oid", this)
|
||||||
val use = Column[PersonUse]("person_use", this)
|
val use = Column[PersonUse]("person_use", this)
|
||||||
val all = NonEmptyList.of[Column[_]](
|
val all = Nel.of[Column[_]](
|
||||||
pid,
|
pid,
|
||||||
cid,
|
cid,
|
||||||
name,
|
name,
|
||||||
@ -122,7 +122,7 @@ object RPerson {
|
|||||||
def findLike(
|
def findLike(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
personName: String,
|
personName: String,
|
||||||
use: NonEmptyList[PersonUse]
|
use: Nel[PersonUse]
|
||||||
): ConnectionIO[Vector[IdRef]] =
|
): ConnectionIO[Vector[IdRef]] =
|
||||||
run(
|
run(
|
||||||
select(T.pid, T.name),
|
select(T.pid, T.name),
|
||||||
@ -134,7 +134,7 @@ object RPerson {
|
|||||||
coll: Ident,
|
coll: Ident,
|
||||||
contactKind: ContactKind,
|
contactKind: ContactKind,
|
||||||
value: String,
|
value: String,
|
||||||
use: NonEmptyList[PersonUse]
|
use: Nel[PersonUse]
|
||||||
): ConnectionIO[Vector[IdRef]] = {
|
): ConnectionIO[Vector[IdRef]] = {
|
||||||
val p = RPerson.as("p")
|
val p = RPerson.as("p")
|
||||||
val c = RContact.as("c")
|
val c = RContact.as("c")
|
||||||
@ -162,7 +162,7 @@ object RPerson {
|
|||||||
def findAllRef(
|
def findAllRef(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
nameQ: Option[String],
|
nameQ: Option[String],
|
||||||
order: Table => Column[_]
|
order: Table => Nel[OrderBy]
|
||||||
): ConnectionIO[Vector[IdRef]] = {
|
): ConnectionIO[Vector[IdRef]] = {
|
||||||
|
|
||||||
val nameFilter = nameQ.map(s => T.name.like(s"%${s.toLowerCase}%"))
|
val nameFilter = nameQ.map(s => T.name.like(s"%${s.toLowerCase}%"))
|
||||||
@ -176,7 +176,7 @@ object RPerson {
|
|||||||
DML.delete(T, T.pid === personId && T.cid === coll)
|
DML.delete(T, T.pid === personId && T.cid === coll)
|
||||||
|
|
||||||
def findOrganization(ids: Set[Ident]): ConnectionIO[Vector[PersonRef]] =
|
def findOrganization(ids: Set[Ident]): ConnectionIO[Vector[PersonRef]] =
|
||||||
NonEmptyList.fromList(ids.toList) match {
|
Nel.fromList(ids.toList) match {
|
||||||
case Some(nel) =>
|
case Some(nel) =>
|
||||||
run(select(T.pid, T.name, T.oid), from(T), T.pid.in(nel))
|
run(select(T.pid, T.name, T.oid), from(T), T.pid.in(nel))
|
||||||
.query[PersonRef]
|
.query[PersonRef]
|
||||||
|
@ -75,10 +75,11 @@ object RTag {
|
|||||||
|
|
||||||
def findAll(
|
def findAll(
|
||||||
coll: Ident,
|
coll: Ident,
|
||||||
nameQ: Option[String],
|
query: Option[String],
|
||||||
order: Table => Column[_]
|
order: Table => NonEmptyList[OrderBy]
|
||||||
): ConnectionIO[Vector[RTag]] = {
|
): ConnectionIO[Vector[RTag]] = {
|
||||||
val nameFilter = nameQ.map(s => T.name.like(s"%${s.toLowerCase}%"))
|
val nameFilter =
|
||||||
|
query.map(_.toLowerCase).map(s => T.name.like(s"%$s%") || T.category.like(s"%$s%"))
|
||||||
val sql =
|
val sql =
|
||||||
Select(select(T.all), from(T), T.cid === coll &&? nameFilter).orderBy(order(T))
|
Select(select(T.all), from(T), T.cid === coll &&? nameFilter).orderBy(order(T))
|
||||||
sql.build.query[RTag].to[Vector]
|
sql.build.query[RTag].to[Vector]
|
||||||
|
@ -216,8 +216,14 @@ import Api.Model.UserList exposing (UserList)
|
|||||||
import Api.Model.UserPass exposing (UserPass)
|
import Api.Model.UserPass exposing (UserPass)
|
||||||
import Api.Model.VersionInfo exposing (VersionInfo)
|
import Api.Model.VersionInfo exposing (VersionInfo)
|
||||||
import Data.ContactType exposing (ContactType)
|
import Data.ContactType exposing (ContactType)
|
||||||
|
import Data.CustomFieldOrder exposing (CustomFieldOrder)
|
||||||
|
import Data.EquipmentOrder exposing (EquipmentOrder)
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.FolderOrder exposing (FolderOrder)
|
||||||
|
import Data.OrganizationOrder exposing (OrganizationOrder)
|
||||||
|
import Data.PersonOrder exposing (PersonOrder)
|
||||||
import Data.Priority exposing (Priority)
|
import Data.Priority exposing (Priority)
|
||||||
|
import Data.TagOrder exposing (TagOrder)
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import File exposing (File)
|
import File exposing (File)
|
||||||
import Http
|
import Http
|
||||||
@ -291,13 +297,15 @@ putCustomValue flags item fieldValue receive =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
getCustomFields : Flags -> String -> (Result Http.Error CustomFieldList -> msg) -> Cmd msg
|
getCustomFields : Flags -> String -> CustomFieldOrder -> (Result Http.Error CustomFieldList -> msg) -> Cmd msg
|
||||||
getCustomFields flags query receive =
|
getCustomFields flags query order receive =
|
||||||
Http2.authGet
|
Http2.authGet
|
||||||
{ url =
|
{ url =
|
||||||
flags.config.baseUrl
|
flags.config.baseUrl
|
||||||
++ "/api/v1/sec/customfield?q="
|
++ "/api/v1/sec/customfield?q="
|
||||||
++ Url.percentEncode query
|
++ Url.percentEncode query
|
||||||
|
++ "&sort="
|
||||||
|
++ Data.CustomFieldOrder.asString order
|
||||||
, account = getAccount flags
|
, account = getAccount flags
|
||||||
, expect = Http.expectJson receive Api.Model.CustomFieldList.decoder
|
, expect = Http.expectJson receive Api.Model.CustomFieldList.decoder
|
||||||
}
|
}
|
||||||
@ -402,13 +410,21 @@ getFolderDetail flags id receive =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
getFolders : Flags -> String -> Bool -> (Result Http.Error FolderList -> msg) -> Cmd msg
|
getFolders :
|
||||||
getFolders flags query owningOnly receive =
|
Flags
|
||||||
|
-> String
|
||||||
|
-> FolderOrder
|
||||||
|
-> Bool
|
||||||
|
-> (Result Http.Error FolderList -> msg)
|
||||||
|
-> Cmd msg
|
||||||
|
getFolders flags query order owningOnly receive =
|
||||||
Http2.authGet
|
Http2.authGet
|
||||||
{ url =
|
{ url =
|
||||||
flags.config.baseUrl
|
flags.config.baseUrl
|
||||||
++ "/api/v1/sec/folder?q="
|
++ "/api/v1/sec/folder?q="
|
||||||
++ Url.percentEncode query
|
++ Url.percentEncode query
|
||||||
|
++ "&sort="
|
||||||
|
++ Data.FolderOrder.asString order
|
||||||
++ (if owningOnly then
|
++ (if owningOnly then
|
||||||
"&owning=true"
|
"&owning=true"
|
||||||
|
|
||||||
@ -1109,10 +1125,15 @@ getContacts flags kind q receive =
|
|||||||
--- Tags
|
--- Tags
|
||||||
|
|
||||||
|
|
||||||
getTags : Flags -> String -> (Result Http.Error TagList -> msg) -> Cmd msg
|
getTags : Flags -> String -> TagOrder -> (Result Http.Error TagList -> msg) -> Cmd msg
|
||||||
getTags flags query receive =
|
getTags flags query order receive =
|
||||||
Http2.authGet
|
Http2.authGet
|
||||||
{ url = flags.config.baseUrl ++ "/api/v1/sec/tag?q=" ++ Url.percentEncode query
|
{ url =
|
||||||
|
flags.config.baseUrl
|
||||||
|
++ "/api/v1/sec/tag?sort="
|
||||||
|
++ Data.TagOrder.asString order
|
||||||
|
++ "&q="
|
||||||
|
++ Url.percentEncode query
|
||||||
, account = getAccount flags
|
, account = getAccount flags
|
||||||
, expect = Http.expectJson receive Api.Model.TagList.decoder
|
, expect = Http.expectJson receive Api.Model.TagList.decoder
|
||||||
}
|
}
|
||||||
@ -1148,10 +1169,15 @@ deleteTag flags tag receive =
|
|||||||
--- Equipments
|
--- Equipments
|
||||||
|
|
||||||
|
|
||||||
getEquipments : Flags -> String -> (Result Http.Error EquipmentList -> msg) -> Cmd msg
|
getEquipments : Flags -> String -> EquipmentOrder -> (Result Http.Error EquipmentList -> msg) -> Cmd msg
|
||||||
getEquipments flags query receive =
|
getEquipments flags query order receive =
|
||||||
Http2.authGet
|
Http2.authGet
|
||||||
{ url = flags.config.baseUrl ++ "/api/v1/sec/equipment?q=" ++ Url.percentEncode query
|
{ url =
|
||||||
|
flags.config.baseUrl
|
||||||
|
++ "/api/v1/sec/equipment?q="
|
||||||
|
++ Url.percentEncode query
|
||||||
|
++ "&sort="
|
||||||
|
++ Data.EquipmentOrder.asString order
|
||||||
, account = getAccount flags
|
, account = getAccount flags
|
||||||
, expect = Http.expectJson receive Api.Model.EquipmentList.decoder
|
, expect = Http.expectJson receive Api.Model.EquipmentList.decoder
|
||||||
}
|
}
|
||||||
@ -1214,10 +1240,20 @@ getOrgFull id flags receive =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
getOrganizations : Flags -> String -> (Result Http.Error OrganizationList -> msg) -> Cmd msg
|
getOrganizations :
|
||||||
getOrganizations flags query receive =
|
Flags
|
||||||
|
-> String
|
||||||
|
-> OrganizationOrder
|
||||||
|
-> (Result Http.Error OrganizationList -> msg)
|
||||||
|
-> Cmd msg
|
||||||
|
getOrganizations flags query order receive =
|
||||||
Http2.authGet
|
Http2.authGet
|
||||||
{ url = flags.config.baseUrl ++ "/api/v1/sec/organization?full=true&q=" ++ Url.percentEncode query
|
{ url =
|
||||||
|
flags.config.baseUrl
|
||||||
|
++ "/api/v1/sec/organization?full=true&q="
|
||||||
|
++ Url.percentEncode query
|
||||||
|
++ "&sort="
|
||||||
|
++ Data.OrganizationOrder.asString order
|
||||||
, account = getAccount flags
|
, account = getAccount flags
|
||||||
, expect = Http.expectJson receive Api.Model.OrganizationList.decoder
|
, expect = Http.expectJson receive Api.Model.OrganizationList.decoder
|
||||||
}
|
}
|
||||||
@ -1271,10 +1307,15 @@ getPersonFull id flags receive =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
getPersons : Flags -> String -> (Result Http.Error PersonList -> msg) -> Cmd msg
|
getPersons : Flags -> String -> PersonOrder -> (Result Http.Error PersonList -> msg) -> Cmd msg
|
||||||
getPersons flags query receive =
|
getPersons flags query order receive =
|
||||||
Http2.authGet
|
Http2.authGet
|
||||||
{ url = flags.config.baseUrl ++ "/api/v1/sec/person?full=true&q=" ++ Url.percentEncode query
|
{ url =
|
||||||
|
flags.config.baseUrl
|
||||||
|
++ "/api/v1/sec/person?full=true&q="
|
||||||
|
++ Url.percentEncode query
|
||||||
|
++ "&sort="
|
||||||
|
++ Data.PersonOrder.asString order
|
||||||
, account = getAccount flags
|
, account = getAccount flags
|
||||||
, expect = Http.expectJson receive Api.Model.PersonList.decoder
|
, expect = Http.expectJson receive Api.Model.PersonList.decoder
|
||||||
}
|
}
|
||||||
|
@ -25,6 +25,7 @@ import Data.CalEvent exposing (CalEvent)
|
|||||||
import Data.DropdownStyle as DS
|
import Data.DropdownStyle as DS
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Data.ListType exposing (ListType)
|
import Data.ListType exposing (ListType)
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
@ -90,7 +91,7 @@ init flags sett =
|
|||||||
Comp.FixedDropdown.init Data.ListType.all
|
Comp.FixedDropdown.init Data.ListType.all
|
||||||
}
|
}
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
[ Api.getTags flags "" GetTagsResp
|
[ Api.getTags flags "" Data.TagOrder.NameAsc GetTagsResp
|
||||||
, Cmd.map ScheduleMsg cec
|
, Cmd.map ScheduleMsg cec
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
@ -21,6 +21,7 @@ import Comp.Basic as B
|
|||||||
import Comp.CustomFieldForm
|
import Comp.CustomFieldForm
|
||||||
import Comp.CustomFieldTable
|
import Comp.CustomFieldTable
|
||||||
import Comp.MenuBar as MB
|
import Comp.MenuBar as MB
|
||||||
|
import Data.CustomFieldOrder exposing (CustomFieldOrder)
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
@ -36,6 +37,7 @@ type alias Model =
|
|||||||
, fields : List CustomField
|
, fields : List CustomField
|
||||||
, query : String
|
, query : String
|
||||||
, loading : Bool
|
, loading : Bool
|
||||||
|
, order : CustomFieldOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -54,13 +56,14 @@ empty =
|
|||||||
, fields = []
|
, fields = []
|
||||||
, query = ""
|
, query = ""
|
||||||
, loading = False
|
, loading = False
|
||||||
|
, order = Data.CustomFieldOrder.LabelAsc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
init : Flags -> ( Model, Cmd Msg )
|
init : Flags -> ( Model, Cmd Msg )
|
||||||
init flags =
|
init flags =
|
||||||
( empty
|
( empty
|
||||||
, Api.getCustomFields flags empty.query CustomFieldListResp
|
, loadFields flags empty
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -68,14 +71,22 @@ init flags =
|
|||||||
--- Update
|
--- Update
|
||||||
|
|
||||||
|
|
||||||
|
loadFields : Flags -> Model -> Cmd Msg
|
||||||
|
loadFields flags model =
|
||||||
|
Api.getCustomFields flags model.query model.order CustomFieldListResp
|
||||||
|
|
||||||
|
|
||||||
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
||||||
update flags msg model =
|
update flags msg model =
|
||||||
case msg of
|
case msg of
|
||||||
TableMsg lm ->
|
TableMsg lm ->
|
||||||
let
|
let
|
||||||
( tm, action ) =
|
( tm, action, maybeOrder ) =
|
||||||
Comp.CustomFieldTable.update lm model.tableModel
|
Comp.CustomFieldTable.update lm model.tableModel
|
||||||
|
|
||||||
|
newOrder =
|
||||||
|
Maybe.withDefault model.order maybeOrder
|
||||||
|
|
||||||
detail =
|
detail =
|
||||||
case action of
|
case action of
|
||||||
Comp.CustomFieldTable.EditAction item ->
|
Comp.CustomFieldTable.EditAction item ->
|
||||||
@ -83,8 +94,22 @@ update flags msg model =
|
|||||||
|
|
||||||
Comp.CustomFieldTable.NoAction ->
|
Comp.CustomFieldTable.NoAction ->
|
||||||
model.detailModel
|
model.detailModel
|
||||||
|
|
||||||
|
newModel =
|
||||||
|
{ model
|
||||||
|
| tableModel = tm
|
||||||
|
, detailModel = detail
|
||||||
|
, order = newOrder
|
||||||
|
}
|
||||||
|
|
||||||
|
( m1, c1 ) =
|
||||||
|
if model.order == newOrder then
|
||||||
|
( newModel, Cmd.none )
|
||||||
|
|
||||||
|
else
|
||||||
|
( newModel, loadFields flags newModel )
|
||||||
in
|
in
|
||||||
( { model | tableModel = tm, detailModel = detail }, Cmd.none )
|
( m1, c1 )
|
||||||
|
|
||||||
DetailMsg lm ->
|
DetailMsg lm ->
|
||||||
case model.detailModel of
|
case model.detailModel of
|
||||||
@ -95,7 +120,7 @@ update flags msg model =
|
|||||||
|
|
||||||
cmd =
|
cmd =
|
||||||
if back then
|
if back then
|
||||||
Api.getCustomFields flags model.query CustomFieldListResp
|
loadFields flags model
|
||||||
|
|
||||||
else
|
else
|
||||||
Cmd.none
|
Cmd.none
|
||||||
@ -118,8 +143,12 @@ update flags msg model =
|
|||||||
( model, Cmd.none )
|
( model, Cmd.none )
|
||||||
|
|
||||||
SetQuery str ->
|
SetQuery str ->
|
||||||
( { model | query = str }
|
let
|
||||||
, Api.getCustomFields flags str CustomFieldListResp
|
newModel =
|
||||||
|
{ model | query = str }
|
||||||
|
in
|
||||||
|
( newModel
|
||||||
|
, loadFields flags newModel
|
||||||
)
|
)
|
||||||
|
|
||||||
CustomFieldListResp (Ok sl) ->
|
CustomFieldListResp (Ok sl) ->
|
||||||
@ -207,6 +236,7 @@ viewTable2 texts model =
|
|||||||
}
|
}
|
||||||
, Html.map TableMsg
|
, Html.map TableMsg
|
||||||
(Comp.CustomFieldTable.view2 texts.fieldTable
|
(Comp.CustomFieldTable.view2 texts.fieldTable
|
||||||
|
model.order
|
||||||
model.tableModel
|
model.tableModel
|
||||||
model.fields
|
model.fields
|
||||||
)
|
)
|
||||||
|
@ -29,6 +29,7 @@ import Api.Model.ItemFieldValue exposing (ItemFieldValue)
|
|||||||
import Comp.CustomFieldInput
|
import Comp.CustomFieldInput
|
||||||
import Comp.FixedDropdown
|
import Comp.FixedDropdown
|
||||||
import Data.CustomFieldChange exposing (CustomFieldChange(..))
|
import Data.CustomFieldChange exposing (CustomFieldChange(..))
|
||||||
|
import Data.CustomFieldOrder
|
||||||
import Data.CustomFieldType
|
import Data.CustomFieldType
|
||||||
import Data.DropdownStyle as DS
|
import Data.DropdownStyle as DS
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
@ -116,7 +117,7 @@ init flags =
|
|||||||
|
|
||||||
initCmd : Flags -> Cmd Msg
|
initCmd : Flags -> Cmd Msg
|
||||||
initCmd flags =
|
initCmd flags =
|
||||||
Api.getCustomFields flags "" CustomFieldResp
|
Api.getCustomFields flags "" Data.CustomFieldOrder.LabelAsc CustomFieldResp
|
||||||
|
|
||||||
|
|
||||||
setValues : List ItemFieldValue -> Msg
|
setValues : List ItemFieldValue -> Msg
|
||||||
|
@ -16,8 +16,10 @@ module Comp.CustomFieldTable exposing
|
|||||||
|
|
||||||
import Api.Model.CustomField exposing (CustomField)
|
import Api.Model.CustomField exposing (CustomField)
|
||||||
import Comp.Basic as B
|
import Comp.Basic as B
|
||||||
|
import Data.CustomFieldOrder exposing (CustomFieldOrder)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
|
import Html.Events exposing (onClick)
|
||||||
import Messages.Comp.CustomFieldTable exposing (Texts)
|
import Messages.Comp.CustomFieldTable exposing (Texts)
|
||||||
import Styles as S
|
import Styles as S
|
||||||
|
|
||||||
@ -28,6 +30,7 @@ type alias Model =
|
|||||||
|
|
||||||
type Msg
|
type Msg
|
||||||
= EditItem CustomField
|
= EditItem CustomField
|
||||||
|
| ToggleOrder CustomFieldOrder
|
||||||
|
|
||||||
|
|
||||||
type Action
|
type Action
|
||||||
@ -35,31 +38,88 @@ type Action
|
|||||||
| EditAction CustomField
|
| EditAction CustomField
|
||||||
|
|
||||||
|
|
||||||
|
type Header
|
||||||
|
= Label
|
||||||
|
| Format
|
||||||
|
|
||||||
|
|
||||||
init : Model
|
init : Model
|
||||||
init =
|
init =
|
||||||
{}
|
{}
|
||||||
|
|
||||||
|
|
||||||
update : Msg -> Model -> ( Model, Action )
|
update : Msg -> Model -> ( Model, Action, Maybe CustomFieldOrder )
|
||||||
update msg model =
|
update msg model =
|
||||||
case msg of
|
case msg of
|
||||||
EditItem item ->
|
EditItem item ->
|
||||||
( model, EditAction item )
|
( model, EditAction item, Nothing )
|
||||||
|
|
||||||
|
ToggleOrder order ->
|
||||||
|
( model, NoAction, Just order )
|
||||||
|
|
||||||
|
|
||||||
|
newOrder : Header -> CustomFieldOrder -> CustomFieldOrder
|
||||||
|
newOrder header current =
|
||||||
|
case ( header, current ) of
|
||||||
|
( Label, Data.CustomFieldOrder.LabelAsc ) ->
|
||||||
|
Data.CustomFieldOrder.LabelDesc
|
||||||
|
|
||||||
|
( Label, _ ) ->
|
||||||
|
Data.CustomFieldOrder.LabelAsc
|
||||||
|
|
||||||
|
( Format, Data.CustomFieldOrder.FormatAsc ) ->
|
||||||
|
Data.CustomFieldOrder.FormatDesc
|
||||||
|
|
||||||
|
( Format, _ ) ->
|
||||||
|
Data.CustomFieldOrder.FormatAsc
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- View2
|
--- View2
|
||||||
|
|
||||||
|
|
||||||
view2 : Texts -> Model -> List CustomField -> Html Msg
|
view2 : Texts -> CustomFieldOrder -> Model -> List CustomField -> Html Msg
|
||||||
view2 texts _ items =
|
view2 texts order _ items =
|
||||||
|
let
|
||||||
|
labelSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.CustomFieldOrder.LabelAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.CustomFieldOrder.LabelDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
formatSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.CustomFieldOrder.FormatAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.CustomFieldOrder.FormatDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-up"
|
||||||
|
in
|
||||||
div []
|
div []
|
||||||
[ table [ class S.tableMain ]
|
[ table [ class S.tableMain ]
|
||||||
[ thead []
|
[ thead []
|
||||||
[ tr []
|
[ tr []
|
||||||
[ th [] []
|
[ th [] []
|
||||||
, th [ class "text-left" ] [ text texts.nameLabel ]
|
, th [ class "text-left" ]
|
||||||
, th [ class "text-left" ] [ text texts.format ]
|
[ a [ href "#", onClick (ToggleOrder <| newOrder Label order) ]
|
||||||
|
[ i [ class labelSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.nameLabel
|
||||||
|
]
|
||||||
|
]
|
||||||
|
, th [ class "text-left" ]
|
||||||
|
[ a [ href "#", onClick (ToggleOrder <| newOrder Format order) ]
|
||||||
|
[ i [ class formatSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.format
|
||||||
|
]
|
||||||
|
]
|
||||||
, th [ class "text-center hidden sm:table-cell" ] [ text texts.usageCount ]
|
, th [ class "text-center hidden sm:table-cell" ] [ text texts.usageCount ]
|
||||||
, th [ class "text-center hidden sm:table-cell" ] [ text texts.basics.created ]
|
, th [ class "text-center hidden sm:table-cell" ] [ text texts.basics.created ]
|
||||||
]
|
]
|
||||||
|
@ -22,6 +22,7 @@ import Comp.EquipmentForm
|
|||||||
import Comp.EquipmentTable
|
import Comp.EquipmentTable
|
||||||
import Comp.MenuBar as MB
|
import Comp.MenuBar as MB
|
||||||
import Comp.YesNoDimmer
|
import Comp.YesNoDimmer
|
||||||
|
import Data.EquipmentOrder exposing (EquipmentOrder)
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
@ -40,6 +41,7 @@ type alias Model =
|
|||||||
, loading : Bool
|
, loading : Bool
|
||||||
, deleteConfirm : Comp.YesNoDimmer.Model
|
, deleteConfirm : Comp.YesNoDimmer.Model
|
||||||
, query : String
|
, query : String
|
||||||
|
, order : EquipmentOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -64,6 +66,7 @@ emptyModel =
|
|||||||
, loading = False
|
, loading = False
|
||||||
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
||||||
, query = ""
|
, query = ""
|
||||||
|
, order = Data.EquipmentOrder.NameAsc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -86,9 +89,12 @@ update flags msg model =
|
|||||||
case msg of
|
case msg of
|
||||||
TableMsg m ->
|
TableMsg m ->
|
||||||
let
|
let
|
||||||
( tm, tc ) =
|
( tm, tc, maybeOrder ) =
|
||||||
Comp.EquipmentTable.update flags m model.tableModel
|
Comp.EquipmentTable.update flags m model.tableModel
|
||||||
|
|
||||||
|
newOrder =
|
||||||
|
Maybe.withDefault model.order maybeOrder
|
||||||
|
|
||||||
( m2, c2 ) =
|
( m2, c2 ) =
|
||||||
( { model
|
( { model
|
||||||
| tableModel = tm
|
| tableModel = tm
|
||||||
@ -99,6 +105,7 @@ update flags msg model =
|
|||||||
|
|
||||||
else
|
else
|
||||||
model.formError
|
model.formError
|
||||||
|
, order = newOrder
|
||||||
}
|
}
|
||||||
, Cmd.map TableMsg tc
|
, Cmd.map TableMsg tc
|
||||||
)
|
)
|
||||||
@ -110,8 +117,15 @@ update flags msg model =
|
|||||||
|
|
||||||
Nothing ->
|
Nothing ->
|
||||||
( m2, Cmd.none )
|
( m2, Cmd.none )
|
||||||
|
|
||||||
|
( m4, c4 ) =
|
||||||
|
if model.order == newOrder then
|
||||||
|
( m3, Cmd.none )
|
||||||
|
|
||||||
|
else
|
||||||
|
update flags LoadEquipments m3
|
||||||
in
|
in
|
||||||
( m3, Cmd.batch [ c2, c3 ] )
|
( m4, Cmd.batch [ c2, c3, c4 ] )
|
||||||
|
|
||||||
FormMsg m ->
|
FormMsg m ->
|
||||||
let
|
let
|
||||||
@ -121,7 +135,7 @@ update flags msg model =
|
|||||||
( { model | formModel = m2 }, Cmd.map FormMsg c2 )
|
( { model | formModel = m2 }, Cmd.map FormMsg c2 )
|
||||||
|
|
||||||
LoadEquipments ->
|
LoadEquipments ->
|
||||||
( { model | loading = True }, Api.getEquipments flags "" EquipmentResp )
|
( { model | loading = True }, Api.getEquipments flags model.query model.order EquipmentResp )
|
||||||
|
|
||||||
EquipmentResp (Ok equipments) ->
|
EquipmentResp (Ok equipments) ->
|
||||||
let
|
let
|
||||||
@ -211,7 +225,7 @@ update flags msg model =
|
|||||||
m =
|
m =
|
||||||
{ model | query = str }
|
{ model | query = str }
|
||||||
in
|
in
|
||||||
( m, Api.getEquipments flags str EquipmentResp )
|
( m, Api.getEquipments flags str model.order EquipmentResp )
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -251,6 +265,7 @@ viewTable2 texts model =
|
|||||||
}
|
}
|
||||||
, Html.map TableMsg
|
, Html.map TableMsg
|
||||||
(Comp.EquipmentTable.view2 texts.equipmentTable
|
(Comp.EquipmentTable.view2 texts.equipmentTable
|
||||||
|
model.order
|
||||||
model.tableModel
|
model.tableModel
|
||||||
)
|
)
|
||||||
, div
|
, div
|
||||||
|
@ -15,10 +15,12 @@ module Comp.EquipmentTable exposing
|
|||||||
|
|
||||||
import Api.Model.Equipment exposing (Equipment)
|
import Api.Model.Equipment exposing (Equipment)
|
||||||
import Comp.Basic as B
|
import Comp.Basic as B
|
||||||
|
import Data.EquipmentOrder exposing (EquipmentOrder)
|
||||||
import Data.EquipmentUse
|
import Data.EquipmentUse
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
|
import Html.Events exposing (onClick)
|
||||||
import Messages.Comp.EquipmentTable exposing (Texts)
|
import Messages.Comp.EquipmentTable exposing (Texts)
|
||||||
import Styles as S
|
import Styles as S
|
||||||
|
|
||||||
@ -40,27 +42,50 @@ type Msg
|
|||||||
= SetEquipments (List Equipment)
|
= SetEquipments (List Equipment)
|
||||||
| Select Equipment
|
| Select Equipment
|
||||||
| Deselect
|
| Deselect
|
||||||
|
| ToggleOrder EquipmentOrder
|
||||||
|
|
||||||
|
|
||||||
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
update : Flags -> Msg -> Model -> ( Model, Cmd Msg, Maybe EquipmentOrder )
|
||||||
update _ msg model =
|
update _ msg model =
|
||||||
case msg of
|
case msg of
|
||||||
SetEquipments list ->
|
SetEquipments list ->
|
||||||
( { model | equips = list, selected = Nothing }, Cmd.none )
|
( { model | equips = list, selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Select equip ->
|
Select equip ->
|
||||||
( { model | selected = Just equip }, Cmd.none )
|
( { model | selected = Just equip }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Deselect ->
|
Deselect ->
|
||||||
( { model | selected = Nothing }, Cmd.none )
|
( { model | selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
|
ToggleOrder order ->
|
||||||
|
( model, Cmd.none, Just order )
|
||||||
|
|
||||||
|
|
||||||
|
newOrder : EquipmentOrder -> EquipmentOrder
|
||||||
|
newOrder current =
|
||||||
|
case current of
|
||||||
|
Data.EquipmentOrder.NameAsc ->
|
||||||
|
Data.EquipmentOrder.NameDesc
|
||||||
|
|
||||||
|
Data.EquipmentOrder.NameDesc ->
|
||||||
|
Data.EquipmentOrder.NameAsc
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- View2
|
--- View2
|
||||||
|
|
||||||
|
|
||||||
view2 : Texts -> Model -> Html Msg
|
view2 : Texts -> EquipmentOrder -> Model -> Html Msg
|
||||||
view2 texts model =
|
view2 texts order model =
|
||||||
|
let
|
||||||
|
nameSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.EquipmentOrder.NameAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.EquipmentOrder.NameDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
in
|
||||||
table [ class S.tableMain ]
|
table [ class S.tableMain ]
|
||||||
[ thead []
|
[ thead []
|
||||||
[ tr []
|
[ tr []
|
||||||
@ -68,7 +93,12 @@ view2 texts model =
|
|||||||
, th [ class "text-left pr-1 md:px-2 w-20" ]
|
, th [ class "text-left pr-1 md:px-2 w-20" ]
|
||||||
[ text texts.use
|
[ text texts.use
|
||||||
]
|
]
|
||||||
, th [ class "text-left" ] [ text texts.basics.name ]
|
, th [ class "text-left" ]
|
||||||
|
[ a [ href "#", onClick (ToggleOrder <| newOrder order) ]
|
||||||
|
[ i [ class nameSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.basics.name
|
||||||
|
]
|
||||||
|
]
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
, tbody []
|
, tbody []
|
||||||
|
@ -24,6 +24,7 @@ import Comp.FolderDetail
|
|||||||
import Comp.FolderTable
|
import Comp.FolderTable
|
||||||
import Comp.MenuBar as MB
|
import Comp.MenuBar as MB
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.FolderOrder exposing (FolderOrder)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
import Http
|
import Http
|
||||||
@ -39,6 +40,7 @@ type alias Model =
|
|||||||
, query : String
|
, query : String
|
||||||
, owningOnly : Bool
|
, owningOnly : Bool
|
||||||
, loading : Bool
|
, loading : Bool
|
||||||
|
, order : FolderOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -62,6 +64,7 @@ empty =
|
|||||||
, query = ""
|
, query = ""
|
||||||
, owningOnly = True
|
, owningOnly = True
|
||||||
, loading = False
|
, loading = False
|
||||||
|
, order = Data.FolderOrder.NameAsc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -70,7 +73,7 @@ init flags =
|
|||||||
( empty
|
( empty
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
[ Api.getUsers flags UserListResp
|
[ Api.getUsers flags UserListResp
|
||||||
, Api.getFolders flags empty.query empty.owningOnly FolderListResp
|
, loadFolders flags empty
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -79,23 +82,41 @@ init flags =
|
|||||||
--- Update
|
--- Update
|
||||||
|
|
||||||
|
|
||||||
|
loadFolders : Flags -> Model -> Cmd Msg
|
||||||
|
loadFolders flags model =
|
||||||
|
Api.getFolders flags model.query model.order model.owningOnly FolderListResp
|
||||||
|
|
||||||
|
|
||||||
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
||||||
update flags msg model =
|
update flags msg model =
|
||||||
case msg of
|
case msg of
|
||||||
TableMsg lm ->
|
TableMsg lm ->
|
||||||
let
|
let
|
||||||
( tm, action ) =
|
( tm, action, maybeOrder ) =
|
||||||
Comp.FolderTable.update lm model.tableModel
|
Comp.FolderTable.update lm model.tableModel
|
||||||
|
|
||||||
cmd =
|
newOrder =
|
||||||
|
Maybe.withDefault model.order maybeOrder
|
||||||
|
|
||||||
|
newModel =
|
||||||
|
{ model | tableModel = tm, order = newOrder }
|
||||||
|
|
||||||
|
detailCmd =
|
||||||
case action of
|
case action of
|
||||||
Comp.FolderTable.EditAction item ->
|
Comp.FolderTable.EditAction item ->
|
||||||
Api.getFolderDetail flags item.id FolderDetailResp
|
Api.getFolderDetail flags item.id FolderDetailResp
|
||||||
|
|
||||||
Comp.FolderTable.NoAction ->
|
Comp.FolderTable.NoAction ->
|
||||||
Cmd.none
|
Cmd.none
|
||||||
|
|
||||||
|
refreshCmd =
|
||||||
|
if model.order == newOrder then
|
||||||
|
Cmd.none
|
||||||
|
|
||||||
|
else
|
||||||
|
loadFolders flags newModel
|
||||||
in
|
in
|
||||||
( { model | tableModel = tm }, cmd )
|
( newModel, Cmd.batch [ detailCmd, refreshCmd ] )
|
||||||
|
|
||||||
DetailMsg lm ->
|
DetailMsg lm ->
|
||||||
case model.detailModel of
|
case model.detailModel of
|
||||||
@ -106,7 +127,7 @@ update flags msg model =
|
|||||||
|
|
||||||
cmd =
|
cmd =
|
||||||
if back then
|
if back then
|
||||||
Api.getFolders flags model.query model.owningOnly FolderListResp
|
loadFolders flags model
|
||||||
|
|
||||||
else
|
else
|
||||||
Cmd.none
|
Cmd.none
|
||||||
@ -129,17 +150,24 @@ update flags msg model =
|
|||||||
( model, Cmd.none )
|
( model, Cmd.none )
|
||||||
|
|
||||||
SetQuery str ->
|
SetQuery str ->
|
||||||
( { model | query = str }
|
let
|
||||||
, Api.getFolders flags str model.owningOnly FolderListResp
|
nm =
|
||||||
|
{ model | query = str }
|
||||||
|
in
|
||||||
|
( nm
|
||||||
|
, loadFolders flags nm
|
||||||
)
|
)
|
||||||
|
|
||||||
ToggleOwningOnly ->
|
ToggleOwningOnly ->
|
||||||
let
|
let
|
||||||
newOwning =
|
newOwning =
|
||||||
not model.owningOnly
|
not model.owningOnly
|
||||||
|
|
||||||
|
nm =
|
||||||
|
{ model | owningOnly = newOwning }
|
||||||
in
|
in
|
||||||
( { model | owningOnly = newOwning }
|
( nm
|
||||||
, Api.getFolders flags model.query newOwning FolderListResp
|
, loadFolders flags nm
|
||||||
)
|
)
|
||||||
|
|
||||||
UserListResp (Ok ul) ->
|
UserListResp (Ok ul) ->
|
||||||
@ -241,6 +269,7 @@ viewTable2 texts model =
|
|||||||
, Html.map TableMsg
|
, Html.map TableMsg
|
||||||
(Comp.FolderTable.view2
|
(Comp.FolderTable.view2
|
||||||
texts.folderTable
|
texts.folderTable
|
||||||
|
model.order
|
||||||
model.tableModel
|
model.tableModel
|
||||||
model.folders
|
model.folders
|
||||||
)
|
)
|
||||||
|
@ -16,8 +16,10 @@ module Comp.FolderTable exposing
|
|||||||
|
|
||||||
import Api.Model.FolderItem exposing (FolderItem)
|
import Api.Model.FolderItem exposing (FolderItem)
|
||||||
import Comp.Basic as B
|
import Comp.Basic as B
|
||||||
|
import Data.FolderOrder exposing (FolderOrder)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
|
import Html.Events exposing (onClick)
|
||||||
import Messages.Comp.FolderTable exposing (Texts)
|
import Messages.Comp.FolderTable exposing (Texts)
|
||||||
import Styles as S
|
import Styles as S
|
||||||
|
|
||||||
@ -28,6 +30,7 @@ type alias Model =
|
|||||||
|
|
||||||
type Msg
|
type Msg
|
||||||
= EditItem FolderItem
|
= EditItem FolderItem
|
||||||
|
| ToggleOrder FolderOrder
|
||||||
|
|
||||||
|
|
||||||
type Action
|
type Action
|
||||||
@ -35,32 +38,87 @@ type Action
|
|||||||
| EditAction FolderItem
|
| EditAction FolderItem
|
||||||
|
|
||||||
|
|
||||||
|
type Header
|
||||||
|
= Name
|
||||||
|
| Owner
|
||||||
|
|
||||||
|
|
||||||
init : Model
|
init : Model
|
||||||
init =
|
init =
|
||||||
{}
|
{}
|
||||||
|
|
||||||
|
|
||||||
update : Msg -> Model -> ( Model, Action )
|
update : Msg -> Model -> ( Model, Action, Maybe FolderOrder )
|
||||||
update msg model =
|
update msg model =
|
||||||
case msg of
|
case msg of
|
||||||
EditItem item ->
|
EditItem item ->
|
||||||
( model, EditAction item )
|
( model, EditAction item, Nothing )
|
||||||
|
|
||||||
|
ToggleOrder order ->
|
||||||
|
( model, NoAction, Just order )
|
||||||
|
|
||||||
|
|
||||||
|
newOrder : Header -> FolderOrder -> FolderOrder
|
||||||
|
newOrder header current =
|
||||||
|
case ( header, current ) of
|
||||||
|
( Name, Data.FolderOrder.NameAsc ) ->
|
||||||
|
Data.FolderOrder.NameDesc
|
||||||
|
|
||||||
|
( Name, _ ) ->
|
||||||
|
Data.FolderOrder.NameAsc
|
||||||
|
|
||||||
|
( Owner, Data.FolderOrder.OwnerAsc ) ->
|
||||||
|
Data.FolderOrder.OwnerDesc
|
||||||
|
|
||||||
|
( Owner, _ ) ->
|
||||||
|
Data.FolderOrder.OwnerAsc
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- View2
|
--- View2
|
||||||
|
|
||||||
|
|
||||||
view2 : Texts -> Model -> List FolderItem -> Html Msg
|
view2 : Texts -> FolderOrder -> Model -> List FolderItem -> Html Msg
|
||||||
view2 texts _ items =
|
view2 texts order _ items =
|
||||||
|
let
|
||||||
|
nameSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.FolderOrder.NameAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.FolderOrder.NameDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
ownerSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.FolderOrder.OwnerAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.FolderOrder.OwnerDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-up"
|
||||||
|
in
|
||||||
table [ class S.tableMain ]
|
table [ class S.tableMain ]
|
||||||
[ thead []
|
[ thead []
|
||||||
[ tr []
|
[ tr []
|
||||||
[ th [ class "w-px whitespace-nowrap pr-1 md:pr-3" ] []
|
[ th [ class "w-px whitespace-nowrap pr-1 md:pr-3" ] []
|
||||||
, th [ class "text-left" ]
|
, th [ class "text-left" ]
|
||||||
[ text texts.basics.name
|
[ a [ href "#", onClick (ToggleOrder <| newOrder Name order) ]
|
||||||
|
[ i [ class nameSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.basics.name
|
||||||
|
]
|
||||||
|
]
|
||||||
|
, th [ class "text-left hidden sm:table-cell" ]
|
||||||
|
[ a [ href "#", onClick (ToggleOrder <| newOrder Owner order) ]
|
||||||
|
[ i [ class ownerSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.owner
|
||||||
|
]
|
||||||
]
|
]
|
||||||
, th [ class "text-left hidden sm:table-cell" ] [ text "Owner" ]
|
|
||||||
, th [ class "text-center" ]
|
, th [ class "text-center" ]
|
||||||
[ span [ class "hidden sm:inline" ]
|
[ span [ class "hidden sm:inline" ]
|
||||||
[ text texts.memberCount
|
[ text texts.memberCount
|
||||||
|
@ -35,10 +35,14 @@ import Comp.Tabs as TB
|
|||||||
import Data.CustomFieldChange exposing (CustomFieldChange(..))
|
import Data.CustomFieldChange exposing (CustomFieldChange(..))
|
||||||
import Data.Direction exposing (Direction)
|
import Data.Direction exposing (Direction)
|
||||||
import Data.DropdownStyle
|
import Data.DropdownStyle
|
||||||
|
import Data.EquipmentOrder
|
||||||
import Data.Fields
|
import Data.Fields
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.FolderOrder
|
||||||
import Data.Icons as Icons
|
import Data.Icons as Icons
|
||||||
|
import Data.PersonOrder
|
||||||
import Data.PersonUse
|
import Data.PersonUse
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import DatePicker exposing (DatePicker)
|
import DatePicker exposing (DatePicker)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
@ -157,11 +161,11 @@ loadModel flags =
|
|||||||
Comp.DatePicker.init
|
Comp.DatePicker.init
|
||||||
in
|
in
|
||||||
Cmd.batch
|
Cmd.batch
|
||||||
[ Api.getTags flags "" GetTagsResp
|
[ Api.getTags flags "" Data.TagOrder.NameAsc GetTagsResp
|
||||||
, Api.getOrgLight flags GetOrgResp
|
, Api.getOrgLight flags GetOrgResp
|
||||||
, Api.getPersons flags "" GetPersonResp
|
, Api.getPersons flags "" Data.PersonOrder.NameAsc GetPersonResp
|
||||||
, Api.getEquipments flags "" GetEquipResp
|
, Api.getEquipments flags "" Data.EquipmentOrder.NameAsc GetEquipResp
|
||||||
, Api.getFolders flags "" False GetFolderResp
|
, Api.getFolders flags "" Data.FolderOrder.NameAsc False GetFolderResp
|
||||||
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
|
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
|
||||||
, Cmd.map ItemDatePickerMsg dpc
|
, Cmd.map ItemDatePickerMsg dpc
|
||||||
, Cmd.map DueDatePickerMsg dpc
|
, Cmd.map DueDatePickerMsg dpc
|
||||||
|
@ -59,10 +59,14 @@ import Comp.PersonForm
|
|||||||
import Comp.SentMails
|
import Comp.SentMails
|
||||||
import Data.CustomFieldChange exposing (CustomFieldChange(..))
|
import Data.CustomFieldChange exposing (CustomFieldChange(..))
|
||||||
import Data.Direction
|
import Data.Direction
|
||||||
|
import Data.EquipmentOrder
|
||||||
import Data.Fields exposing (Field)
|
import Data.Fields exposing (Field)
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.FolderOrder
|
||||||
import Data.ItemNav exposing (ItemNav)
|
import Data.ItemNav exposing (ItemNav)
|
||||||
|
import Data.PersonOrder
|
||||||
import Data.PersonUse
|
import Data.PersonUse
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import DatePicker
|
import DatePicker
|
||||||
import Dict
|
import Dict
|
||||||
@ -265,7 +269,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
|
, Api.getPersons flags "" Data.PersonOrder.NameAsc GetPersonResp
|
||||||
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
|
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
|
||||||
]
|
]
|
||||||
, sub =
|
, sub =
|
||||||
@ -1642,11 +1646,11 @@ update key flags inav settings msg model =
|
|||||||
getOptions : Flags -> Cmd Msg
|
getOptions : Flags -> Cmd Msg
|
||||||
getOptions flags =
|
getOptions flags =
|
||||||
Cmd.batch
|
Cmd.batch
|
||||||
[ Api.getTags flags "" GetTagsResp
|
[ Api.getTags flags "" Data.TagOrder.NameAsc GetTagsResp
|
||||||
, Api.getOrgLight flags GetOrgResp
|
, Api.getOrgLight flags GetOrgResp
|
||||||
, Api.getPersons flags "" GetPersonResp
|
, Api.getPersons flags "" Data.PersonOrder.NameAsc GetPersonResp
|
||||||
, Api.getEquipments flags "" GetEquipResp
|
, Api.getEquipments flags "" Data.EquipmentOrder.NameAsc GetEquipResp
|
||||||
, Api.getFolders flags "" False GetFolderResp
|
, Api.getFolders flags "" Data.FolderOrder.NameAsc False GetFolderResp
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@ -30,6 +30,7 @@ import Comp.YesNoDimmer
|
|||||||
import Data.CalEvent exposing (CalEvent)
|
import Data.CalEvent exposing (CalEvent)
|
||||||
import Data.DropdownStyle as DS
|
import Data.DropdownStyle as DS
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Data.Validated exposing (Validated(..))
|
import Data.Validated exposing (Validated(..))
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
@ -182,7 +183,7 @@ init flags =
|
|||||||
}
|
}
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
[ Api.getMailSettings flags "" ConnResp
|
[ Api.getMailSettings flags "" ConnResp
|
||||||
, Api.getTags flags "" GetTagsResp
|
, Api.getTags flags "" Data.TagOrder.NameAsc GetTagsResp
|
||||||
, Cmd.map CalEventMsg scmd
|
, Cmd.map CalEventMsg scmd
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
@ -23,6 +23,7 @@ import Comp.OrgForm
|
|||||||
import Comp.OrgTable
|
import Comp.OrgTable
|
||||||
import Comp.YesNoDimmer
|
import Comp.YesNoDimmer
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.OrganizationOrder exposing (OrganizationOrder)
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
@ -41,6 +42,7 @@ type alias Model =
|
|||||||
, loading : Bool
|
, loading : Bool
|
||||||
, deleteConfirm : Comp.YesNoDimmer.Model
|
, deleteConfirm : Comp.YesNoDimmer.Model
|
||||||
, query : String
|
, query : String
|
||||||
|
, order : OrganizationOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -65,6 +67,7 @@ emptyModel =
|
|||||||
, loading = False
|
, loading = False
|
||||||
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
||||||
, query = ""
|
, query = ""
|
||||||
|
, order = Data.OrganizationOrder.NameAsc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -87,7 +90,7 @@ update flags msg model =
|
|||||||
case msg of
|
case msg of
|
||||||
TableMsg m ->
|
TableMsg m ->
|
||||||
let
|
let
|
||||||
( tm, tc ) =
|
( tm, tc, maybeOrder ) =
|
||||||
Comp.OrgTable.update flags m model.tableModel
|
Comp.OrgTable.update flags m model.tableModel
|
||||||
|
|
||||||
( m2, c2 ) =
|
( m2, c2 ) =
|
||||||
@ -100,6 +103,7 @@ update flags msg model =
|
|||||||
|
|
||||||
else
|
else
|
||||||
model.formError
|
model.formError
|
||||||
|
, order = Maybe.withDefault model.order maybeOrder
|
||||||
}
|
}
|
||||||
, Cmd.map TableMsg tc
|
, Cmd.map TableMsg tc
|
||||||
)
|
)
|
||||||
@ -111,8 +115,15 @@ update flags msg model =
|
|||||||
|
|
||||||
Nothing ->
|
Nothing ->
|
||||||
( m2, Cmd.none )
|
( m2, Cmd.none )
|
||||||
|
|
||||||
|
( m4, c4 ) =
|
||||||
|
if maybeOrder /= Nothing && maybeOrder /= Just model.order then
|
||||||
|
update flags LoadOrgs m3
|
||||||
|
|
||||||
|
else
|
||||||
|
( m3, Cmd.none )
|
||||||
in
|
in
|
||||||
( m3, Cmd.batch [ c2, c3 ] )
|
( m4, Cmd.batch [ c2, c3, c4 ] )
|
||||||
|
|
||||||
FormMsg m ->
|
FormMsg m ->
|
||||||
let
|
let
|
||||||
@ -122,7 +133,13 @@ update flags msg model =
|
|||||||
( { model | formModel = m2 }, Cmd.map FormMsg c2 )
|
( { model | formModel = m2 }, Cmd.map FormMsg c2 )
|
||||||
|
|
||||||
LoadOrgs ->
|
LoadOrgs ->
|
||||||
( { model | loading = True }, Api.getOrganizations flags model.query OrgResp )
|
( { model | loading = True }
|
||||||
|
, Api.getOrganizations
|
||||||
|
flags
|
||||||
|
model.query
|
||||||
|
model.order
|
||||||
|
OrgResp
|
||||||
|
)
|
||||||
|
|
||||||
OrgResp (Ok orgs) ->
|
OrgResp (Ok orgs) ->
|
||||||
let
|
let
|
||||||
@ -212,7 +229,7 @@ update flags msg model =
|
|||||||
m =
|
m =
|
||||||
{ model | query = str }
|
{ model | query = str }
|
||||||
in
|
in
|
||||||
( m, Api.getOrganizations flags str OrgResp )
|
( m, Api.getOrganizations flags str model.order OrgResp )
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -250,7 +267,7 @@ viewTable2 texts model =
|
|||||||
]
|
]
|
||||||
, rootClasses = "mb-4"
|
, rootClasses = "mb-4"
|
||||||
}
|
}
|
||||||
, Html.map TableMsg (Comp.OrgTable.view2 texts.orgTable model.tableModel)
|
, Html.map TableMsg (Comp.OrgTable.view2 texts.orgTable model.order model.tableModel)
|
||||||
, B.loadingDimmer
|
, B.loadingDimmer
|
||||||
{ active = model.loading
|
{ active = model.loading
|
||||||
, label = texts.basics.loading
|
, label = texts.basics.loading
|
||||||
|
@ -17,8 +17,10 @@ import Api.Model.Organization exposing (Organization)
|
|||||||
import Comp.Basic as B
|
import Comp.Basic as B
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Data.OrgUse
|
import Data.OrgUse
|
||||||
|
import Data.OrganizationOrder exposing (OrganizationOrder)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
|
import Html.Events exposing (onClick)
|
||||||
import Messages.Comp.OrgTable exposing (Texts)
|
import Messages.Comp.OrgTable exposing (Texts)
|
||||||
import Styles as S
|
import Styles as S
|
||||||
import Util.Address
|
import Util.Address
|
||||||
@ -42,27 +44,50 @@ type Msg
|
|||||||
= SetOrgs (List Organization)
|
= SetOrgs (List Organization)
|
||||||
| Select Organization
|
| Select Organization
|
||||||
| Deselect
|
| Deselect
|
||||||
|
| ToggleOrder OrganizationOrder
|
||||||
|
|
||||||
|
|
||||||
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
update : Flags -> Msg -> Model -> ( Model, Cmd Msg, Maybe OrganizationOrder )
|
||||||
update _ msg model =
|
update _ msg model =
|
||||||
case msg of
|
case msg of
|
||||||
SetOrgs list ->
|
SetOrgs list ->
|
||||||
( { model | orgs = list, selected = Nothing }, Cmd.none )
|
( { model | orgs = list, selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Select equip ->
|
Select equip ->
|
||||||
( { model | selected = Just equip }, Cmd.none )
|
( { model | selected = Just equip }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Deselect ->
|
Deselect ->
|
||||||
( { model | selected = Nothing }, Cmd.none )
|
( { model | selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
|
ToggleOrder order ->
|
||||||
|
( model, Cmd.none, Just order )
|
||||||
|
|
||||||
|
|
||||||
|
newOrder : OrganizationOrder -> OrganizationOrder
|
||||||
|
newOrder current =
|
||||||
|
case current of
|
||||||
|
Data.OrganizationOrder.NameAsc ->
|
||||||
|
Data.OrganizationOrder.NameDesc
|
||||||
|
|
||||||
|
Data.OrganizationOrder.NameDesc ->
|
||||||
|
Data.OrganizationOrder.NameAsc
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- View2
|
--- View2
|
||||||
|
|
||||||
|
|
||||||
view2 : Texts -> Model -> Html Msg
|
view2 : Texts -> OrganizationOrder -> Model -> Html Msg
|
||||||
view2 texts model =
|
view2 texts order model =
|
||||||
|
let
|
||||||
|
nameSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.OrganizationOrder.NameAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.OrganizationOrder.NameDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
in
|
||||||
table [ class S.tableMain ]
|
table [ class S.tableMain ]
|
||||||
[ thead []
|
[ thead []
|
||||||
[ tr []
|
[ tr []
|
||||||
@ -71,7 +96,10 @@ view2 texts model =
|
|||||||
[ text texts.use
|
[ text texts.use
|
||||||
]
|
]
|
||||||
, th [ class "text-left" ]
|
, th [ class "text-left" ]
|
||||||
[ text texts.basics.name
|
[ a [ href "#", onClick (ToggleOrder <| newOrder order) ]
|
||||||
|
[ i [ class nameSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.basics.name
|
||||||
|
]
|
||||||
]
|
]
|
||||||
, th [ class "text-left hidden md:table-cell" ]
|
, th [ class "text-left hidden md:table-cell" ]
|
||||||
[ text texts.address
|
[ text texts.address
|
||||||
|
@ -24,6 +24,7 @@ import Comp.PersonForm
|
|||||||
import Comp.PersonTable
|
import Comp.PersonTable
|
||||||
import Comp.YesNoDimmer
|
import Comp.YesNoDimmer
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.PersonOrder exposing (PersonOrder)
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
@ -42,6 +43,7 @@ type alias Model =
|
|||||||
, loading : Int
|
, loading : Int
|
||||||
, deleteConfirm : Comp.YesNoDimmer.Model
|
, deleteConfirm : Comp.YesNoDimmer.Model
|
||||||
, query : String
|
, query : String
|
||||||
|
, order : PersonOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -66,6 +68,7 @@ emptyModel =
|
|||||||
, loading = 0
|
, loading = 0
|
||||||
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
||||||
, query = ""
|
, query = ""
|
||||||
|
, order = Data.PersonOrder.NameAsc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -89,9 +92,12 @@ update flags msg model =
|
|||||||
case msg of
|
case msg of
|
||||||
TableMsg m ->
|
TableMsg m ->
|
||||||
let
|
let
|
||||||
( tm, tc ) =
|
( tm, tc, maybeOrder ) =
|
||||||
Comp.PersonTable.update flags m model.tableModel
|
Comp.PersonTable.update flags m model.tableModel
|
||||||
|
|
||||||
|
newOrder =
|
||||||
|
Maybe.withDefault model.order maybeOrder
|
||||||
|
|
||||||
( m2, c2 ) =
|
( m2, c2 ) =
|
||||||
( { model
|
( { model
|
||||||
| tableModel = tm
|
| tableModel = tm
|
||||||
@ -102,6 +108,7 @@ update flags msg model =
|
|||||||
|
|
||||||
else
|
else
|
||||||
model.formError
|
model.formError
|
||||||
|
, order = newOrder
|
||||||
}
|
}
|
||||||
, Cmd.map TableMsg tc
|
, Cmd.map TableMsg tc
|
||||||
)
|
)
|
||||||
@ -113,8 +120,15 @@ update flags msg model =
|
|||||||
|
|
||||||
Nothing ->
|
Nothing ->
|
||||||
( m2, Cmd.none )
|
( m2, Cmd.none )
|
||||||
|
|
||||||
|
( m4, c4 ) =
|
||||||
|
if model.order == newOrder then
|
||||||
|
( m3, Cmd.none )
|
||||||
|
|
||||||
|
else
|
||||||
|
update flags LoadPersons m3
|
||||||
in
|
in
|
||||||
( m3, Cmd.batch [ c2, c3 ] )
|
( m4, Cmd.batch [ c2, c3, c4 ] )
|
||||||
|
|
||||||
FormMsg m ->
|
FormMsg m ->
|
||||||
let
|
let
|
||||||
@ -126,7 +140,7 @@ update flags msg model =
|
|||||||
LoadPersons ->
|
LoadPersons ->
|
||||||
( { model | loading = model.loading + 2 }
|
( { model | loading = model.loading + 2 }
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
[ Api.getPersons flags model.query PersonResp
|
[ Api.getPersons flags model.query model.order PersonResp
|
||||||
, Api.getOrgLight flags GetOrgResp
|
, Api.getOrgLight flags GetOrgResp
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
@ -244,7 +258,7 @@ update flags msg model =
|
|||||||
m =
|
m =
|
||||||
{ model | query = str }
|
{ model | query = str }
|
||||||
in
|
in
|
||||||
( m, Api.getPersons flags str PersonResp )
|
( m, Api.getPersons flags str model.order PersonResp )
|
||||||
|
|
||||||
|
|
||||||
isLoading : Model -> Bool
|
isLoading : Model -> Bool
|
||||||
@ -287,7 +301,7 @@ viewTable2 texts model =
|
|||||||
]
|
]
|
||||||
, rootClasses = "mb-4"
|
, rootClasses = "mb-4"
|
||||||
}
|
}
|
||||||
, Html.map TableMsg (Comp.PersonTable.view2 texts.personTable model.tableModel)
|
, Html.map TableMsg (Comp.PersonTable.view2 texts.personTable model.order model.tableModel)
|
||||||
, B.loadingDimmer
|
, B.loadingDimmer
|
||||||
{ active = isLoading model
|
{ active = isLoading model
|
||||||
, label = texts.basics.loading
|
, label = texts.basics.loading
|
||||||
|
@ -16,9 +16,11 @@ module Comp.PersonTable exposing
|
|||||||
import Api.Model.Person exposing (Person)
|
import Api.Model.Person exposing (Person)
|
||||||
import Comp.Basic as B
|
import Comp.Basic as B
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.PersonOrder exposing (PersonOrder)
|
||||||
import Data.PersonUse
|
import Data.PersonUse
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
|
import Html.Events exposing (onClick)
|
||||||
import Messages.Comp.PersonTable exposing (Texts)
|
import Messages.Comp.PersonTable exposing (Texts)
|
||||||
import Styles as S
|
import Styles as S
|
||||||
import Util.Contact
|
import Util.Contact
|
||||||
@ -41,27 +43,75 @@ type Msg
|
|||||||
= SetPersons (List Person)
|
= SetPersons (List Person)
|
||||||
| Select Person
|
| Select Person
|
||||||
| Deselect
|
| Deselect
|
||||||
|
| ToggleOrder PersonOrder
|
||||||
|
|
||||||
|
|
||||||
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
update : Flags -> Msg -> Model -> ( Model, Cmd Msg, Maybe PersonOrder )
|
||||||
update _ msg model =
|
update _ msg model =
|
||||||
case msg of
|
case msg of
|
||||||
SetPersons list ->
|
SetPersons list ->
|
||||||
( { model | equips = list, selected = Nothing }, Cmd.none )
|
( { model | equips = list, selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Select equip ->
|
Select equip ->
|
||||||
( { model | selected = Just equip }, Cmd.none )
|
( { model | selected = Just equip }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Deselect ->
|
Deselect ->
|
||||||
( { model | selected = Nothing }, Cmd.none )
|
( { model | selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
|
ToggleOrder order ->
|
||||||
|
( model, Cmd.none, Just order )
|
||||||
|
|
||||||
|
|
||||||
|
type Header
|
||||||
|
= Name
|
||||||
|
| Org
|
||||||
|
|
||||||
|
|
||||||
|
newOrder : Header -> PersonOrder -> PersonOrder
|
||||||
|
newOrder header current =
|
||||||
|
case ( header, current ) of
|
||||||
|
( Name, Data.PersonOrder.NameAsc ) ->
|
||||||
|
Data.PersonOrder.NameDesc
|
||||||
|
|
||||||
|
( Name, _ ) ->
|
||||||
|
Data.PersonOrder.NameAsc
|
||||||
|
|
||||||
|
( Org, Data.PersonOrder.OrgAsc ) ->
|
||||||
|
Data.PersonOrder.OrgDesc
|
||||||
|
|
||||||
|
( Org, _ ) ->
|
||||||
|
Data.PersonOrder.OrgAsc
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- View2
|
--- View2
|
||||||
|
|
||||||
|
|
||||||
view2 : Texts -> Model -> Html Msg
|
view2 : Texts -> PersonOrder -> Model -> Html Msg
|
||||||
view2 texts model =
|
view2 texts order model =
|
||||||
|
let
|
||||||
|
nameSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.PersonOrder.NameAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.PersonOrder.NameDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
orgSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.PersonOrder.OrgAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.PersonOrder.OrgDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-down-alt"
|
||||||
|
in
|
||||||
table [ class S.tableMain ]
|
table [ class S.tableMain ]
|
||||||
[ thead []
|
[ thead []
|
||||||
[ tr []
|
[ tr []
|
||||||
@ -69,8 +119,18 @@ view2 texts model =
|
|||||||
, th [ class "text-left pr-1 md:px-2" ]
|
, th [ class "text-left pr-1 md:px-2" ]
|
||||||
[ text texts.use
|
[ text texts.use
|
||||||
]
|
]
|
||||||
, th [ class "text-left" ] [ text texts.basics.name ]
|
, th [ class "text-left" ]
|
||||||
, th [ class "text-left hidden sm:table-cell" ] [ text texts.basics.organization ]
|
[ a [ href "#", onClick (ToggleOrder <| newOrder Name order) ]
|
||||||
|
[ i [ class nameSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.basics.name
|
||||||
|
]
|
||||||
|
]
|
||||||
|
, th [ class "text-left hidden sm:table-cell" ]
|
||||||
|
[ a [ href "#", onClick (ToggleOrder <| newOrder Org order) ]
|
||||||
|
[ i [ class orgSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.basics.organization
|
||||||
|
]
|
||||||
|
]
|
||||||
, th [ class "text-left hidden md:table-cell" ] [ text texts.contact ]
|
, th [ class "text-left hidden md:table-cell" ] [ text texts.contact ]
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
|
@ -37,7 +37,9 @@ import Data.CalEvent exposing (CalEvent)
|
|||||||
import Data.Direction exposing (Direction(..))
|
import Data.Direction exposing (Direction(..))
|
||||||
import Data.DropdownStyle as DS
|
import Data.DropdownStyle as DS
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.FolderOrder
|
||||||
import Data.Language exposing (Language)
|
import Data.Language exposing (Language)
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Data.Validated exposing (Validated(..))
|
import Data.Validated exposing (Validated(..))
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
@ -221,8 +223,8 @@ initWith flags s =
|
|||||||
[ Api.getImapSettings flags "" ConnResp
|
[ Api.getImapSettings flags "" ConnResp
|
||||||
, nc
|
, nc
|
||||||
, Cmd.map CalEventMsg sc
|
, Cmd.map CalEventMsg sc
|
||||||
, Api.getFolders flags "" False GetFolderResp
|
, Api.getFolders flags "" Data.FolderOrder.NameAsc False GetFolderResp
|
||||||
, Api.getTags flags "" GetTagResp
|
, Api.getTags flags "" Data.TagOrder.NameAsc GetTagResp
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -268,8 +270,8 @@ init flags =
|
|||||||
}
|
}
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
[ Api.getImapSettings flags "" ConnResp
|
[ Api.getImapSettings flags "" ConnResp
|
||||||
, Api.getFolders flags "" False GetFolderResp
|
, Api.getFolders flags "" Data.FolderOrder.NameAsc False GetFolderResp
|
||||||
, Api.getTags flags "" GetTagResp
|
, Api.getTags flags "" Data.TagOrder.NameAsc GetTagResp
|
||||||
, Cmd.map CalEventMsg scmd
|
, Cmd.map CalEventMsg scmd
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
@ -40,10 +40,12 @@ import Comp.TagSelect
|
|||||||
import Data.CustomFieldChange exposing (CustomFieldValueCollect)
|
import Data.CustomFieldChange exposing (CustomFieldValueCollect)
|
||||||
import Data.Direction exposing (Direction)
|
import Data.Direction exposing (Direction)
|
||||||
import Data.DropdownStyle as DS
|
import Data.DropdownStyle as DS
|
||||||
|
import Data.EquipmentOrder
|
||||||
import Data.EquipmentUse
|
import Data.EquipmentUse
|
||||||
import Data.Fields
|
import Data.Fields
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Data.ItemQuery as Q exposing (ItemQuery)
|
import Data.ItemQuery as Q exposing (ItemQuery)
|
||||||
|
import Data.PersonOrder
|
||||||
import Data.PersonUse
|
import Data.PersonUse
|
||||||
import Data.SearchMode exposing (SearchMode)
|
import Data.SearchMode exposing (SearchMode)
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
@ -441,8 +443,8 @@ updateDrop ddm flags settings msg model =
|
|||||||
Cmd.batch
|
Cmd.batch
|
||||||
[ Api.itemSearchStats flags Api.Model.ItemQuery.empty GetAllTagsResp
|
[ Api.itemSearchStats flags Api.Model.ItemQuery.empty GetAllTagsResp
|
||||||
, Api.getOrgLight flags GetOrgResp
|
, Api.getOrgLight flags GetOrgResp
|
||||||
, Api.getEquipments flags "" GetEquipResp
|
, Api.getEquipments flags "" Data.EquipmentOrder.NameAsc GetEquipResp
|
||||||
, Api.getPersons flags "" GetPersonResp
|
, Api.getPersons flags "" Data.PersonOrder.NameAsc GetPersonResp
|
||||||
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
|
, Cmd.map CustomFieldMsg (Comp.CustomFieldMultiInput.initCmd flags)
|
||||||
, cdp
|
, cdp
|
||||||
]
|
]
|
||||||
@ -1097,6 +1099,7 @@ tabLook settings model tab =
|
|||||||
hiddenOr fields default =
|
hiddenOr fields default =
|
||||||
if List.all isHidden fields then
|
if List.all isHidden fields then
|
||||||
Comp.Tabs.Hidden
|
Comp.Tabs.Hidden
|
||||||
|
|
||||||
else
|
else
|
||||||
default
|
default
|
||||||
|
|
||||||
@ -1179,7 +1182,6 @@ searchTabState settings model tab =
|
|||||||
searchTab =
|
searchTab =
|
||||||
findTab tab
|
findTab tab
|
||||||
|
|
||||||
|
|
||||||
folded =
|
folded =
|
||||||
if Set.member tab.name model.openTabs then
|
if Set.member tab.name model.openTabs then
|
||||||
Comp.Tabs.Open
|
Comp.Tabs.Open
|
||||||
@ -1189,10 +1191,10 @@ searchTabState settings model tab =
|
|||||||
|
|
||||||
state =
|
state =
|
||||||
{ folded = folded
|
{ folded = folded
|
||||||
, look = Maybe.map (tabLook settings model) searchTab
|
, look =
|
||||||
|
Maybe.map (tabLook settings model) searchTab
|
||||||
|> Maybe.withDefault Comp.Tabs.Normal
|
|> Maybe.withDefault Comp.Tabs.Normal
|
||||||
}
|
}
|
||||||
|
|
||||||
in
|
in
|
||||||
( state, ToggleAkkordionTab tab.name )
|
( state, ToggleAkkordionTab tab.name )
|
||||||
|
|
||||||
|
@ -27,8 +27,10 @@ import Comp.Dropdown exposing (isDropdownChangeMsg)
|
|||||||
import Comp.FixedDropdown
|
import Comp.FixedDropdown
|
||||||
import Data.DropdownStyle as DS
|
import Data.DropdownStyle as DS
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.FolderOrder
|
||||||
import Data.Language exposing (Language)
|
import Data.Language exposing (Language)
|
||||||
import Data.Priority exposing (Priority)
|
import Data.Priority exposing (Priority)
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
@ -89,8 +91,8 @@ init : Flags -> ( Model, Cmd Msg )
|
|||||||
init flags =
|
init flags =
|
||||||
( emptyModel
|
( emptyModel
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
[ Api.getFolders flags "" False GetFolderResp
|
[ Api.getFolders flags "" Data.FolderOrder.NameAsc False GetFolderResp
|
||||||
, Api.getTags flags "" GetTagResp
|
, Api.getTags flags "" Data.TagOrder.NameAsc GetTagResp
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -23,6 +23,7 @@ import Comp.TagForm
|
|||||||
import Comp.TagTable
|
import Comp.TagTable
|
||||||
import Comp.YesNoDimmer
|
import Comp.YesNoDimmer
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.TagOrder exposing (TagOrder)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
import Html.Events exposing (onSubmit)
|
import Html.Events exposing (onSubmit)
|
||||||
@ -42,6 +43,7 @@ type alias Model =
|
|||||||
, loading : Bool
|
, loading : Bool
|
||||||
, deleteConfirm : Comp.YesNoDimmer.Model
|
, deleteConfirm : Comp.YesNoDimmer.Model
|
||||||
, query : String
|
, query : String
|
||||||
|
, order : TagOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -66,6 +68,7 @@ emptyModel =
|
|||||||
, loading = False
|
, loading = False
|
||||||
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
, deleteConfirm = Comp.YesNoDimmer.emptyModel
|
||||||
, query = ""
|
, query = ""
|
||||||
|
, order = Data.TagOrder.NameAsc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -88,12 +91,16 @@ update flags msg model =
|
|||||||
case msg of
|
case msg of
|
||||||
TableMsg m ->
|
TableMsg m ->
|
||||||
let
|
let
|
||||||
( tm, tc ) =
|
( tm, tc, maybeOrder ) =
|
||||||
Comp.TagTable.update flags m model.tagTableModel
|
Comp.TagTable.update flags m model.tagTableModel
|
||||||
|
|
||||||
|
newOrder =
|
||||||
|
Maybe.withDefault model.order maybeOrder
|
||||||
|
|
||||||
( m2, c2 ) =
|
( m2, c2 ) =
|
||||||
( { model
|
( { model
|
||||||
| tagTableModel = tm
|
| tagTableModel = tm
|
||||||
|
, order = newOrder
|
||||||
, viewMode = Maybe.map (\_ -> Form) tm.selected |> Maybe.withDefault Table
|
, viewMode = Maybe.map (\_ -> Form) tm.selected |> Maybe.withDefault Table
|
||||||
, formError =
|
, formError =
|
||||||
if Util.Maybe.nonEmpty tm.selected then
|
if Util.Maybe.nonEmpty tm.selected then
|
||||||
@ -112,8 +119,15 @@ update flags msg model =
|
|||||||
|
|
||||||
Nothing ->
|
Nothing ->
|
||||||
( m2, Cmd.none )
|
( m2, Cmd.none )
|
||||||
|
|
||||||
|
( m4, c4 ) =
|
||||||
|
if model.order == newOrder then
|
||||||
|
( m3, Cmd.none )
|
||||||
|
|
||||||
|
else
|
||||||
|
update flags LoadTags m3
|
||||||
in
|
in
|
||||||
( m3, Cmd.batch [ c2, c3 ] )
|
( m4, Cmd.batch [ c2, c3, c4 ] )
|
||||||
|
|
||||||
FormMsg m ->
|
FormMsg m ->
|
||||||
let
|
let
|
||||||
@ -123,7 +137,9 @@ update flags msg model =
|
|||||||
( { model | tagFormModel = m2 }, Cmd.map FormMsg c2 )
|
( { model | tagFormModel = m2 }, Cmd.map FormMsg c2 )
|
||||||
|
|
||||||
LoadTags ->
|
LoadTags ->
|
||||||
( { model | loading = True }, Api.getTags flags model.query (TagResp model.query) )
|
( { model | loading = True }
|
||||||
|
, Api.getTags flags model.query model.order (TagResp model.query)
|
||||||
|
)
|
||||||
|
|
||||||
TagResp query (Ok tags) ->
|
TagResp query (Ok tags) ->
|
||||||
let
|
let
|
||||||
@ -224,7 +240,7 @@ update flags msg model =
|
|||||||
m =
|
m =
|
||||||
{ model | query = str }
|
{ model | query = str }
|
||||||
in
|
in
|
||||||
( m, Api.getTags flags str (TagResp str) )
|
( m, Api.getTags flags str model.order (TagResp str) )
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -262,7 +278,7 @@ viewTable2 texts model =
|
|||||||
]
|
]
|
||||||
, rootClasses = "mb-4"
|
, rootClasses = "mb-4"
|
||||||
}
|
}
|
||||||
, Html.map TableMsg (Comp.TagTable.view2 texts.tagTable model.tagTableModel)
|
, Html.map TableMsg (Comp.TagTable.view2 texts.tagTable model.order model.tagTableModel)
|
||||||
, div
|
, div
|
||||||
[ classList
|
[ classList
|
||||||
[ ( "ui dimmer", True )
|
[ ( "ui dimmer", True )
|
||||||
|
@ -16,8 +16,10 @@ module Comp.TagTable exposing
|
|||||||
import Api.Model.Tag exposing (Tag)
|
import Api.Model.Tag exposing (Tag)
|
||||||
import Comp.Basic as B
|
import Comp.Basic as B
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
|
import Data.TagOrder exposing (TagOrder)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
|
import Html.Events exposing (onClick)
|
||||||
import Messages.Comp.TagTable exposing (Texts)
|
import Messages.Comp.TagTable exposing (Texts)
|
||||||
import Styles as S
|
import Styles as S
|
||||||
|
|
||||||
@ -35,37 +37,108 @@ emptyModel =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
type Header
|
||||||
|
= Name
|
||||||
|
| Category
|
||||||
|
|
||||||
|
|
||||||
type Msg
|
type Msg
|
||||||
= SetTags (List Tag)
|
= SetTags (List Tag)
|
||||||
| Select Tag
|
| Select Tag
|
||||||
| Deselect
|
| Deselect
|
||||||
|
| SortClick TagOrder
|
||||||
|
|
||||||
|
|
||||||
update : Flags -> Msg -> Model -> ( Model, Cmd Msg )
|
update : Flags -> Msg -> Model -> ( Model, Cmd Msg, Maybe TagOrder )
|
||||||
update _ msg model =
|
update _ msg model =
|
||||||
case msg of
|
case msg of
|
||||||
SetTags list ->
|
SetTags list ->
|
||||||
( { model | tags = list, selected = Nothing }, Cmd.none )
|
( { model | tags = list, selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Select tag ->
|
Select tag ->
|
||||||
( { model | selected = Just tag }, Cmd.none )
|
( { model | selected = Just tag }, Cmd.none, Nothing )
|
||||||
|
|
||||||
Deselect ->
|
Deselect ->
|
||||||
( { model | selected = Nothing }, Cmd.none )
|
( { model | selected = Nothing }, Cmd.none, Nothing )
|
||||||
|
|
||||||
|
SortClick order ->
|
||||||
|
( model, Cmd.none, Just order )
|
||||||
|
|
||||||
|
|
||||||
|
newOrder : Header -> TagOrder -> TagOrder
|
||||||
|
newOrder header current =
|
||||||
|
case ( header, current ) of
|
||||||
|
( Name, Data.TagOrder.NameAsc ) ->
|
||||||
|
Data.TagOrder.NameDesc
|
||||||
|
|
||||||
|
( Name, Data.TagOrder.NameDesc ) ->
|
||||||
|
Data.TagOrder.NameAsc
|
||||||
|
|
||||||
|
( Name, Data.TagOrder.CategoryAsc ) ->
|
||||||
|
Data.TagOrder.NameAsc
|
||||||
|
|
||||||
|
( Name, Data.TagOrder.CategoryDesc ) ->
|
||||||
|
Data.TagOrder.NameAsc
|
||||||
|
|
||||||
|
( Category, Data.TagOrder.NameAsc ) ->
|
||||||
|
Data.TagOrder.CategoryAsc
|
||||||
|
|
||||||
|
( Category, Data.TagOrder.NameDesc ) ->
|
||||||
|
Data.TagOrder.CategoryAsc
|
||||||
|
|
||||||
|
( Category, Data.TagOrder.CategoryAsc ) ->
|
||||||
|
Data.TagOrder.CategoryDesc
|
||||||
|
|
||||||
|
( Category, Data.TagOrder.CategoryDesc ) ->
|
||||||
|
Data.TagOrder.CategoryAsc
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- View2
|
--- View2
|
||||||
|
|
||||||
|
|
||||||
view2 : Texts -> Model -> Html Msg
|
view2 : Texts -> TagOrder -> Model -> Html Msg
|
||||||
view2 texts model =
|
view2 texts order model =
|
||||||
|
let
|
||||||
|
nameSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.TagOrder.NameAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.TagOrder.NameDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-down"
|
||||||
|
|
||||||
|
catSortIcon =
|
||||||
|
case order of
|
||||||
|
Data.TagOrder.CategoryAsc ->
|
||||||
|
"fa fa-sort-alpha-up"
|
||||||
|
|
||||||
|
Data.TagOrder.CategoryDesc ->
|
||||||
|
"fa fa-sort-alpha-down-alt"
|
||||||
|
|
||||||
|
_ ->
|
||||||
|
"invisible fa fa-sort-alpha-down"
|
||||||
|
in
|
||||||
table [ class S.tableMain ]
|
table [ class S.tableMain ]
|
||||||
[ thead []
|
[ thead []
|
||||||
[ tr []
|
[ tr []
|
||||||
[ th [ class "" ] []
|
[ th [ class "" ] []
|
||||||
, th [ class "text-left" ] [ text texts.basics.name ]
|
, th [ class "text-left" ]
|
||||||
, th [ class "text-left" ] [ text texts.category ]
|
[ a [ href "#", onClick (SortClick <| newOrder Name order) ]
|
||||||
|
[ i [ class nameSortIcon, class "mr-1" ] []
|
||||||
|
, text texts.basics.name
|
||||||
|
]
|
||||||
|
]
|
||||||
|
, th [ class "text-left" ]
|
||||||
|
[ a [ href "#", onClick (SortClick <| newOrder Category order) ]
|
||||||
|
[ i [ class catSortIcon, class "mr-1" ]
|
||||||
|
[]
|
||||||
|
, text texts.category
|
||||||
|
]
|
||||||
|
]
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
, tbody []
|
, tbody []
|
||||||
|
@ -28,6 +28,7 @@ import Data.DropdownStyle as DS
|
|||||||
import Data.Fields exposing (Field)
|
import Data.Fields exposing (Field)
|
||||||
import Data.Flags exposing (Flags)
|
import Data.Flags exposing (Flags)
|
||||||
import Data.ItemTemplate as IT exposing (ItemTemplate)
|
import Data.ItemTemplate as IT exposing (ItemTemplate)
|
||||||
|
import Data.TagOrder
|
||||||
import Data.UiSettings exposing (ItemPattern, Pos(..), UiSettings)
|
import Data.UiSettings exposing (ItemPattern, Pos(..), UiSettings)
|
||||||
import Dict exposing (Dict)
|
import Dict exposing (Dict)
|
||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
@ -160,7 +161,7 @@ init flags settings =
|
|||||||
Comp.FixedDropdown.init Messages.UiLanguage.all
|
Comp.FixedDropdown.init Messages.UiLanguage.all
|
||||||
, openTabs = Set.empty
|
, openTabs = Set.empty
|
||||||
}
|
}
|
||||||
, Api.getTags flags "" GetTagsResp
|
, Api.getTags flags "" Data.TagOrder.NameAsc GetTagsResp
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
31
modules/webapp/src/main/elm/Data/CustomFieldOrder.elm
Normal file
31
modules/webapp/src/main/elm/Data/CustomFieldOrder.elm
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{-
|
||||||
|
Copyright 2020 Docspell Contributors
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
-}
|
||||||
|
|
||||||
|
|
||||||
|
module Data.CustomFieldOrder exposing (CustomFieldOrder(..), asString)
|
||||||
|
|
||||||
|
|
||||||
|
type CustomFieldOrder
|
||||||
|
= LabelAsc
|
||||||
|
| LabelDesc
|
||||||
|
| FormatAsc
|
||||||
|
| FormatDesc
|
||||||
|
|
||||||
|
|
||||||
|
asString : CustomFieldOrder -> String
|
||||||
|
asString order =
|
||||||
|
case order of
|
||||||
|
LabelAsc ->
|
||||||
|
"label"
|
||||||
|
|
||||||
|
LabelDesc ->
|
||||||
|
"-label"
|
||||||
|
|
||||||
|
FormatAsc ->
|
||||||
|
"type"
|
||||||
|
|
||||||
|
FormatDesc ->
|
||||||
|
"-type"
|
23
modules/webapp/src/main/elm/Data/EquipmentOrder.elm
Normal file
23
modules/webapp/src/main/elm/Data/EquipmentOrder.elm
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
{-
|
||||||
|
Copyright 2020 Docspell Contributors
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
-}
|
||||||
|
|
||||||
|
|
||||||
|
module Data.EquipmentOrder exposing (EquipmentOrder(..), asString)
|
||||||
|
|
||||||
|
|
||||||
|
type EquipmentOrder
|
||||||
|
= NameAsc
|
||||||
|
| NameDesc
|
||||||
|
|
||||||
|
|
||||||
|
asString : EquipmentOrder -> String
|
||||||
|
asString order =
|
||||||
|
case order of
|
||||||
|
NameAsc ->
|
||||||
|
"name"
|
||||||
|
|
||||||
|
NameDesc ->
|
||||||
|
"-name"
|
31
modules/webapp/src/main/elm/Data/FolderOrder.elm
Normal file
31
modules/webapp/src/main/elm/Data/FolderOrder.elm
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{-
|
||||||
|
Copyright 2020 Docspell Contributors
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
-}
|
||||||
|
|
||||||
|
|
||||||
|
module Data.FolderOrder exposing (FolderOrder(..), asString)
|
||||||
|
|
||||||
|
|
||||||
|
type FolderOrder
|
||||||
|
= NameAsc
|
||||||
|
| NameDesc
|
||||||
|
| OwnerAsc
|
||||||
|
| OwnerDesc
|
||||||
|
|
||||||
|
|
||||||
|
asString : FolderOrder -> String
|
||||||
|
asString order =
|
||||||
|
case order of
|
||||||
|
NameAsc ->
|
||||||
|
"name"
|
||||||
|
|
||||||
|
NameDesc ->
|
||||||
|
"-name"
|
||||||
|
|
||||||
|
OwnerAsc ->
|
||||||
|
"owner"
|
||||||
|
|
||||||
|
OwnerDesc ->
|
||||||
|
"-owner"
|
23
modules/webapp/src/main/elm/Data/OrganizationOrder.elm
Normal file
23
modules/webapp/src/main/elm/Data/OrganizationOrder.elm
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
{-
|
||||||
|
Copyright 2020 Docspell Contributors
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
-}
|
||||||
|
|
||||||
|
|
||||||
|
module Data.OrganizationOrder exposing (OrganizationOrder(..), asString)
|
||||||
|
|
||||||
|
|
||||||
|
type OrganizationOrder
|
||||||
|
= NameAsc
|
||||||
|
| NameDesc
|
||||||
|
|
||||||
|
|
||||||
|
asString : OrganizationOrder -> String
|
||||||
|
asString order =
|
||||||
|
case order of
|
||||||
|
NameAsc ->
|
||||||
|
"name"
|
||||||
|
|
||||||
|
NameDesc ->
|
||||||
|
"-name"
|
31
modules/webapp/src/main/elm/Data/PersonOrder.elm
Normal file
31
modules/webapp/src/main/elm/Data/PersonOrder.elm
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{-
|
||||||
|
Copyright 2020 Docspell Contributors
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
-}
|
||||||
|
|
||||||
|
|
||||||
|
module Data.PersonOrder exposing (PersonOrder(..), asString)
|
||||||
|
|
||||||
|
|
||||||
|
type PersonOrder
|
||||||
|
= NameAsc
|
||||||
|
| NameDesc
|
||||||
|
| OrgAsc
|
||||||
|
| OrgDesc
|
||||||
|
|
||||||
|
|
||||||
|
asString : PersonOrder -> String
|
||||||
|
asString order =
|
||||||
|
case order of
|
||||||
|
NameAsc ->
|
||||||
|
"name"
|
||||||
|
|
||||||
|
NameDesc ->
|
||||||
|
"-name"
|
||||||
|
|
||||||
|
OrgAsc ->
|
||||||
|
"org"
|
||||||
|
|
||||||
|
OrgDesc ->
|
||||||
|
"-org"
|
31
modules/webapp/src/main/elm/Data/TagOrder.elm
Normal file
31
modules/webapp/src/main/elm/Data/TagOrder.elm
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{-
|
||||||
|
Copyright 2020 Docspell Contributors
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
-}
|
||||||
|
|
||||||
|
|
||||||
|
module Data.TagOrder exposing (TagOrder(..), asString)
|
||||||
|
|
||||||
|
|
||||||
|
type TagOrder
|
||||||
|
= NameAsc
|
||||||
|
| NameDesc
|
||||||
|
| CategoryAsc
|
||||||
|
| CategoryDesc
|
||||||
|
|
||||||
|
|
||||||
|
asString : TagOrder -> String
|
||||||
|
asString order =
|
||||||
|
case order of
|
||||||
|
NameAsc ->
|
||||||
|
"name"
|
||||||
|
|
||||||
|
NameDesc ->
|
||||||
|
"-name"
|
||||||
|
|
||||||
|
CategoryAsc ->
|
||||||
|
"category"
|
||||||
|
|
||||||
|
CategoryDesc ->
|
||||||
|
"-category"
|
@ -20,6 +20,7 @@ type alias Texts =
|
|||||||
{ basics : Messages.Basics.Texts
|
{ basics : Messages.Basics.Texts
|
||||||
, memberCount : String
|
, memberCount : String
|
||||||
, formatDateShort : Int -> String
|
, formatDateShort : Int -> String
|
||||||
|
, owner : String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -28,6 +29,7 @@ gb =
|
|||||||
{ basics = Messages.Basics.gb
|
{ basics = Messages.Basics.gb
|
||||||
, memberCount = "#Member"
|
, memberCount = "#Member"
|
||||||
, formatDateShort = DF.formatDateShort Messages.UiLanguage.English
|
, formatDateShort = DF.formatDateShort Messages.UiLanguage.English
|
||||||
|
, owner = "Owner"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -36,4 +38,5 @@ de =
|
|||||||
{ basics = Messages.Basics.de
|
{ basics = Messages.Basics.de
|
||||||
, memberCount = "#Mitglieder"
|
, memberCount = "#Mitglieder"
|
||||||
, formatDateShort = DF.formatDateShort Messages.UiLanguage.German
|
, formatDateShort = DF.formatDateShort Messages.UiLanguage.German
|
||||||
|
, owner = "Besitzer"
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user