mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 19:09:32 +00:00
Merge pull request #768 from stefan-scheidewig/docspell-626-grouped_deletion_of_attachments
Docspell-626: Grouped deletion of attachments
This commit is contained in:
commit
9f322e667d
@ -1,18 +1,16 @@
|
|||||||
package docspell.backend.ops
|
package docspell.backend.ops
|
||||||
|
|
||||||
import cats.data.NonEmptyList
|
import cats.data.{NonEmptyList, OptionT}
|
||||||
import cats.data.OptionT
|
|
||||||
import cats.effect.{Effect, Resource}
|
import cats.effect.{Effect, Resource}
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
import docspell.backend.JobFactory
|
import docspell.backend.JobFactory
|
||||||
import docspell.common._
|
import docspell.common._
|
||||||
import docspell.ftsclient.FtsClient
|
import docspell.ftsclient.FtsClient
|
||||||
import docspell.store.UpdateResult
|
|
||||||
import docspell.store.queries.{QAttachment, QItem, QMoveAttachment}
|
import docspell.store.queries.{QAttachment, QItem, QMoveAttachment}
|
||||||
import docspell.store.queue.JobQueue
|
import docspell.store.queue.JobQueue
|
||||||
import docspell.store.records._
|
import docspell.store.records._
|
||||||
import docspell.store.{AddResult, Store}
|
import docspell.store.{AddResult, Store, UpdateResult}
|
||||||
|
|
||||||
import doobie.implicits._
|
import doobie.implicits._
|
||||||
import org.log4s.getLogger
|
import org.log4s.getLogger
|
||||||
@ -140,6 +138,11 @@ trait OItem[F[_]] {
|
|||||||
|
|
||||||
def deleteAttachment(id: Ident, collective: Ident): F[Int]
|
def deleteAttachment(id: Ident, collective: Ident): F[Int]
|
||||||
|
|
||||||
|
def deleteAttachmentMultiple(
|
||||||
|
attachments: NonEmptyList[Ident],
|
||||||
|
collective: Ident
|
||||||
|
): F[Int]
|
||||||
|
|
||||||
def moveAttachmentBefore(itemId: Ident, source: Ident, target: Ident): F[AddResult]
|
def moveAttachmentBefore(itemId: Ident, source: Ident, target: Ident): F[AddResult]
|
||||||
|
|
||||||
def setAttachmentName(
|
def setAttachmentName(
|
||||||
@ -602,6 +605,20 @@ object OItem {
|
|||||||
.deleteSingleAttachment(store)(id, collective)
|
.deleteSingleAttachment(store)(id, collective)
|
||||||
.flatTap(_ => fts.removeAttachment(logger, id))
|
.flatTap(_ => fts.removeAttachment(logger, id))
|
||||||
|
|
||||||
|
def deleteAttachmentMultiple(
|
||||||
|
attachments: NonEmptyList[Ident],
|
||||||
|
collective: Ident
|
||||||
|
): F[Int] =
|
||||||
|
for {
|
||||||
|
attachmentIds <- store.transact(
|
||||||
|
RAttachment.filterAttachments(attachments, collective)
|
||||||
|
)
|
||||||
|
results <- attachmentIds.traverse(attachment =>
|
||||||
|
deleteAttachment(attachment, collective)
|
||||||
|
)
|
||||||
|
n = results.sum
|
||||||
|
} yield n
|
||||||
|
|
||||||
def setAttachmentName(
|
def setAttachmentName(
|
||||||
attachId: Ident,
|
attachId: Ident,
|
||||||
name: Option[String],
|
name: Option[String],
|
||||||
|
@ -2789,6 +2789,28 @@ paths:
|
|||||||
schema:
|
schema:
|
||||||
$ref: "#/components/schemas/BasicResult"
|
$ref: "#/components/schemas/BasicResult"
|
||||||
|
|
||||||
|
/sec/attachments/delete:
|
||||||
|
post:
|
||||||
|
tags:
|
||||||
|
- Attachment (Multi Edit)
|
||||||
|
summary: Delete multiple attachments.
|
||||||
|
description: |
|
||||||
|
Given a list of attachment ids, deletes all of them.
|
||||||
|
security:
|
||||||
|
- authTokenHeader: [ ]
|
||||||
|
requestBody:
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: "#/components/schemas/IdList"
|
||||||
|
responses:
|
||||||
|
200:
|
||||||
|
description: Ok
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: "#/components/schemas/BasicResult"
|
||||||
|
|
||||||
/sec/queue/state:
|
/sec/queue/state:
|
||||||
get:
|
get:
|
||||||
tags: [ Job Queue ]
|
tags: [ Job Queue ]
|
||||||
|
@ -80,6 +80,7 @@ object RestServer {
|
|||||||
"item" -> ItemRoutes(cfg, pools.blocker, restApp.backend, token),
|
"item" -> ItemRoutes(cfg, pools.blocker, restApp.backend, token),
|
||||||
"items" -> ItemMultiRoutes(restApp.backend, token),
|
"items" -> ItemMultiRoutes(restApp.backend, token),
|
||||||
"attachment" -> AttachmentRoutes(pools.blocker, restApp.backend, token),
|
"attachment" -> AttachmentRoutes(pools.blocker, restApp.backend, token),
|
||||||
|
"attachments" -> AttachmentMultiRoutes(restApp.backend, token),
|
||||||
"upload" -> UploadRoutes.secured(restApp.backend, cfg, token),
|
"upload" -> UploadRoutes.secured(restApp.backend, cfg, token),
|
||||||
"checkfile" -> CheckFileRoutes.secured(restApp.backend, token),
|
"checkfile" -> CheckFileRoutes.secured(restApp.backend, token),
|
||||||
"email/send" -> MailSendRoutes(restApp.backend, token),
|
"email/send" -> MailSendRoutes(restApp.backend, token),
|
||||||
|
@ -0,0 +1,33 @@
|
|||||||
|
package docspell.restserver.conv
|
||||||
|
|
||||||
|
import cats.data.NonEmptyList
|
||||||
|
import cats.implicits._
|
||||||
|
import cats.{ApplicativeError, MonadError}
|
||||||
|
|
||||||
|
import docspell.common.Ident
|
||||||
|
|
||||||
|
import io.circe.DecodingFailure
|
||||||
|
|
||||||
|
trait MultiIdSupport {
|
||||||
|
|
||||||
|
protected def readId[F[_]](
|
||||||
|
id: String
|
||||||
|
)(implicit F: ApplicativeError[F, Throwable]): F[Ident] =
|
||||||
|
Ident
|
||||||
|
.fromString(id)
|
||||||
|
.fold(
|
||||||
|
err => F.raiseError(DecodingFailure(err, Nil)),
|
||||||
|
F.pure
|
||||||
|
)
|
||||||
|
|
||||||
|
protected def readIds[F[_]](ids: List[String])(implicit
|
||||||
|
F: MonadError[F, Throwable]
|
||||||
|
): F[NonEmptyList[Ident]] =
|
||||||
|
ids.traverse(readId[F]).map(NonEmptyList.fromList).flatMap {
|
||||||
|
case Some(nel) => nel.pure[F]
|
||||||
|
case None =>
|
||||||
|
F.raiseError(
|
||||||
|
DecodingFailure("Empty list found, at least one element required", Nil)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,40 @@
|
|||||||
|
package docspell.restserver.routes
|
||||||
|
|
||||||
|
import cats.effect.Effect
|
||||||
|
import cats.implicits._
|
||||||
|
|
||||||
|
import docspell.backend.BackendApp
|
||||||
|
import docspell.backend.auth.AuthToken
|
||||||
|
import docspell.restapi.model._
|
||||||
|
import docspell.restserver.conv.MultiIdSupport
|
||||||
|
|
||||||
|
import org.http4s.HttpRoutes
|
||||||
|
import org.http4s.circe.CirceEntityDecoder._
|
||||||
|
import org.http4s.circe.CirceEntityEncoder._
|
||||||
|
import org.http4s.dsl.Http4sDsl
|
||||||
|
|
||||||
|
object AttachmentMultiRoutes extends MultiIdSupport {
|
||||||
|
|
||||||
|
def apply[F[_]: Effect](
|
||||||
|
backend: BackendApp[F],
|
||||||
|
user: AuthToken
|
||||||
|
): HttpRoutes[F] = {
|
||||||
|
|
||||||
|
val dsl = new Http4sDsl[F] {}
|
||||||
|
import dsl._
|
||||||
|
|
||||||
|
HttpRoutes.of { case req @ POST -> Root / "delete" =>
|
||||||
|
for {
|
||||||
|
json <- req.as[IdList]
|
||||||
|
attachments <- readIds[F](json.ids)
|
||||||
|
n <- backend.item.deleteAttachmentMultiple(attachments, user.account.collective)
|
||||||
|
res = BasicResult(
|
||||||
|
n > 0,
|
||||||
|
if (n > 0) "Attachment(s) deleted" else "Attachment deletion failed."
|
||||||
|
)
|
||||||
|
resp <- Ok(res)
|
||||||
|
} yield resp
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,25 +1,21 @@
|
|||||||
package docspell.restserver.routes
|
package docspell.restserver.routes
|
||||||
|
|
||||||
import cats.ApplicativeError
|
|
||||||
import cats.MonadError
|
|
||||||
import cats.data.NonEmptyList
|
|
||||||
import cats.effect._
|
import cats.effect._
|
||||||
import cats.implicits._
|
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.{RemoveValue, SetValue}
|
import docspell.backend.ops.OCustomFields.{RemoveValue, SetValue}
|
||||||
import docspell.common.{Ident, ItemState}
|
import docspell.common.ItemState
|
||||||
import docspell.restapi.model._
|
import docspell.restapi.model._
|
||||||
import docspell.restserver.conv.Conversions
|
import docspell.restserver.conv.{Conversions, MultiIdSupport}
|
||||||
|
|
||||||
import io.circe.DecodingFailure
|
|
||||||
import org.http4s.HttpRoutes
|
import org.http4s.HttpRoutes
|
||||||
import org.http4s.circe.CirceEntityDecoder._
|
import org.http4s.circe.CirceEntityDecoder._
|
||||||
import org.http4s.circe.CirceEntityEncoder._
|
import org.http4s.circe.CirceEntityEncoder._
|
||||||
import org.http4s.dsl.Http4sDsl
|
import org.http4s.dsl.Http4sDsl
|
||||||
|
|
||||||
object ItemMultiRoutes {
|
object ItemMultiRoutes extends MultiIdSupport {
|
||||||
|
|
||||||
def apply[F[_]: Effect](
|
def apply[F[_]: Effect](
|
||||||
backend: BackendApp[F],
|
backend: BackendApp[F],
|
||||||
@ -215,25 +211,4 @@ object ItemMultiRoutes {
|
|||||||
def notEmpty: Option[String] =
|
def notEmpty: Option[String] =
|
||||||
Option(str).notEmpty
|
Option(str).notEmpty
|
||||||
}
|
}
|
||||||
|
|
||||||
private def readId[F[_]](
|
|
||||||
id: String
|
|
||||||
)(implicit F: ApplicativeError[F, Throwable]): F[Ident] =
|
|
||||||
Ident
|
|
||||||
.fromString(id)
|
|
||||||
.fold(
|
|
||||||
err => F.raiseError(DecodingFailure(err, Nil)),
|
|
||||||
F.pure
|
|
||||||
)
|
|
||||||
|
|
||||||
private def readIds[F[_]](ids: List[String])(implicit
|
|
||||||
F: MonadError[F, Throwable]
|
|
||||||
): F[NonEmptyList[Ident]] =
|
|
||||||
ids.traverse(readId[F]).map(NonEmptyList.fromList).flatMap {
|
|
||||||
case Some(nel) => nel.pure[F]
|
|
||||||
case None =>
|
|
||||||
F.raiseError(
|
|
||||||
DecodingFailure("Empty list found, at least one element required", Nil)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -301,4 +301,19 @@ object RAttachment {
|
|||||||
coll.map(cid => i.cid === cid)
|
coll.map(cid => i.cid === cid)
|
||||||
).build.query[RAttachment].streamWithChunkSize(chunkSize)
|
).build.query[RAttachment].streamWithChunkSize(chunkSize)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def filterAttachments(
|
||||||
|
attachments: NonEmptyList[Ident],
|
||||||
|
coll: Ident
|
||||||
|
): ConnectionIO[Vector[Ident]] = {
|
||||||
|
val a = RAttachment.as("a")
|
||||||
|
val i = RItem.as("i")
|
||||||
|
|
||||||
|
Select(
|
||||||
|
select(a.id),
|
||||||
|
from(a)
|
||||||
|
.innerJoin(i, i.id === a.itemId),
|
||||||
|
i.cid === coll && a.id.in(attachments)
|
||||||
|
).build.query[Ident].to[Vector]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,7 @@ module Api exposing
|
|||||||
, createScanMailbox
|
, createScanMailbox
|
||||||
, deleteAllItems
|
, deleteAllItems
|
||||||
, deleteAttachment
|
, deleteAttachment
|
||||||
|
, deleteAttachments
|
||||||
, deleteCustomField
|
, deleteCustomField
|
||||||
, deleteCustomValue
|
, deleteCustomValue
|
||||||
, deleteCustomValueMultiple
|
, deleteCustomValueMultiple
|
||||||
@ -611,6 +612,24 @@ deleteAttachment flags attachId receive =
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
--- Delete Attachments
|
||||||
|
|
||||||
|
|
||||||
|
deleteAttachments :
|
||||||
|
Flags
|
||||||
|
-> Set String
|
||||||
|
-> (Result Http.Error BasicResult -> msg)
|
||||||
|
-> Cmd msg
|
||||||
|
deleteAttachments flags attachIds receive =
|
||||||
|
Http2.authPost
|
||||||
|
{ url = flags.config.baseUrl ++ "/api/v1/sec/attachments/delete"
|
||||||
|
, account = getAccount flags
|
||||||
|
, body = Http.jsonBody (Api.Model.IdList.encode (Set.toList attachIds |> IdList))
|
||||||
|
, expect = Http.expectJson receive Api.Model.BasicResult.decoder
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- Attachment Metadata
|
--- Attachment Metadata
|
||||||
|
|
||||||
|
|
||||||
|
@ -4,8 +4,11 @@ module Comp.ItemDetail.Model exposing
|
|||||||
, Msg(..)
|
, Msg(..)
|
||||||
, NotesField(..)
|
, NotesField(..)
|
||||||
, SaveNameState(..)
|
, SaveNameState(..)
|
||||||
|
, SelectActionMode(..)
|
||||||
, UpdateResult
|
, UpdateResult
|
||||||
|
, ViewMode(..)
|
||||||
, emptyModel
|
, emptyModel
|
||||||
|
, initSelectViewModel
|
||||||
, isEditNotes
|
, isEditNotes
|
||||||
, personMatchesOrg
|
, personMatchesOrg
|
||||||
, resultModel
|
, resultModel
|
||||||
@ -105,9 +108,26 @@ type alias Model =
|
|||||||
, allPersons : Dict String Person
|
, allPersons : Dict String Person
|
||||||
, attachmentDropdownOpen : Bool
|
, attachmentDropdownOpen : Bool
|
||||||
, editMenuTabsOpen : Set String
|
, editMenuTabsOpen : Set String
|
||||||
|
, viewMode : ViewMode
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
type ViewMode
|
||||||
|
= SimpleView
|
||||||
|
| SelectView SelectViewModel
|
||||||
|
|
||||||
|
|
||||||
|
type alias SelectViewModel =
|
||||||
|
{ ids : Set String
|
||||||
|
, action : SelectActionMode
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
type SelectActionMode
|
||||||
|
= NoneAction
|
||||||
|
| DeleteSelected
|
||||||
|
|
||||||
|
|
||||||
type NotesField
|
type NotesField
|
||||||
= ViewNotes
|
= ViewNotes
|
||||||
| EditNotes Comp.MarkdownInput.Model
|
| EditNotes Comp.MarkdownInput.Model
|
||||||
@ -185,6 +205,14 @@ emptyModel =
|
|||||||
, allPersons = Dict.empty
|
, allPersons = Dict.empty
|
||||||
, attachmentDropdownOpen = False
|
, attachmentDropdownOpen = False
|
||||||
, editMenuTabsOpen = Set.empty
|
, editMenuTabsOpen = Set.empty
|
||||||
|
, viewMode = SimpleView
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
initSelectViewModel : SelectViewModel
|
||||||
|
initSelectViewModel =
|
||||||
|
{ ids = Set.empty
|
||||||
|
, action = NoneAction
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -194,6 +222,7 @@ type Msg
|
|||||||
| Init
|
| Init
|
||||||
| SetItem ItemDetail
|
| SetItem ItemDetail
|
||||||
| SetActiveAttachment Int
|
| SetActiveAttachment Int
|
||||||
|
| ToggleAttachment String
|
||||||
| TagDropdownMsg (Comp.Dropdown.Msg Tag)
|
| TagDropdownMsg (Comp.Dropdown.Msg Tag)
|
||||||
| DirDropdownMsg (Comp.Dropdown.Msg Direction)
|
| DirDropdownMsg (Comp.Dropdown.Msg Direction)
|
||||||
| OrgDropdownMsg (Comp.Dropdown.Msg IdName)
|
| OrgDropdownMsg (Comp.Dropdown.Msg IdName)
|
||||||
@ -239,6 +268,8 @@ type Msg
|
|||||||
| TogglePdfNativeView Bool
|
| TogglePdfNativeView Bool
|
||||||
| RequestDeleteAttachment String
|
| RequestDeleteAttachment String
|
||||||
| DeleteAttachConfirmed String
|
| DeleteAttachConfirmed String
|
||||||
|
| RequestDeleteSelected
|
||||||
|
| DeleteSelectedConfirmed
|
||||||
| AttachModalCancelled
|
| AttachModalCancelled
|
||||||
| DeleteAttachResp (Result Http.Error BasicResult)
|
| DeleteAttachResp (Result Http.Error BasicResult)
|
||||||
| AddFilesToggle
|
| AddFilesToggle
|
||||||
@ -283,6 +314,7 @@ type Msg
|
|||||||
| ReprocessFileResp (Result Http.Error BasicResult)
|
| ReprocessFileResp (Result Http.Error BasicResult)
|
||||||
| RequestReprocessItem
|
| RequestReprocessItem
|
||||||
| ReprocessItemConfirmed
|
| ReprocessItemConfirmed
|
||||||
|
| ToggleSelectView
|
||||||
|
|
||||||
|
|
||||||
type SaveNameState
|
type SaveNameState
|
||||||
|
@ -4,13 +4,7 @@ import Api
|
|||||||
import Api.Model.Attachment exposing (Attachment)
|
import Api.Model.Attachment exposing (Attachment)
|
||||||
import Comp.AttachmentMeta
|
import Comp.AttachmentMeta
|
||||||
import Comp.ConfirmModal
|
import Comp.ConfirmModal
|
||||||
import Comp.ItemDetail.Model
|
import Comp.ItemDetail.Model exposing (Model, Msg(..), NotesField(..), SaveNameState(..), ViewMode(..))
|
||||||
exposing
|
|
||||||
( Model
|
|
||||||
, Msg(..)
|
|
||||||
, NotesField(..)
|
|
||||||
, SaveNameState(..)
|
|
||||||
)
|
|
||||||
import Comp.MenuBar as MB
|
import Comp.MenuBar as MB
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import Dict
|
import Dict
|
||||||
@ -19,7 +13,7 @@ import Html.Attributes exposing (..)
|
|||||||
import Html.Events exposing (onClick, onInput)
|
import Html.Events exposing (onClick, onInput)
|
||||||
import Html5.DragDrop as DD
|
import Html5.DragDrop as DD
|
||||||
import Messages.Comp.ItemDetail.SingleAttachment exposing (Texts)
|
import Messages.Comp.ItemDetail.SingleAttachment exposing (Texts)
|
||||||
import Page exposing (Page(..))
|
import Set
|
||||||
import Styles as S
|
import Styles as S
|
||||||
import Util.Maybe
|
import Util.Maybe
|
||||||
import Util.Size
|
import Util.Size
|
||||||
@ -87,6 +81,7 @@ view texts settings model pos attach =
|
|||||||
- toggle thumbs
|
- toggle thumbs
|
||||||
- name + size
|
- name + size
|
||||||
- eye icon to open it
|
- eye icon to open it
|
||||||
|
- toggle multi select
|
||||||
- menu
|
- menu
|
||||||
- rename
|
- rename
|
||||||
- meta data
|
- meta data
|
||||||
@ -112,6 +107,28 @@ attachHeader texts settings model _ attach =
|
|||||||
multiAttach =
|
multiAttach =
|
||||||
List.length model.item.attachments > 1
|
List.length model.item.attachments > 1
|
||||||
|
|
||||||
|
selectPossible =
|
||||||
|
multiAttach && model.attachMenuOpen
|
||||||
|
|
||||||
|
selectView =
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView _ ->
|
||||||
|
True
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
False
|
||||||
|
|
||||||
|
selectToggleText =
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView _ ->
|
||||||
|
texts.exitSelectMode
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
texts.selectModeTitle
|
||||||
|
|
||||||
|
noAttachmentsSelected =
|
||||||
|
List.isEmpty model.item.attachments
|
||||||
|
|
||||||
attachSelectToggle mobile =
|
attachSelectToggle mobile =
|
||||||
a
|
a
|
||||||
[ href "#"
|
[ href "#"
|
||||||
@ -143,15 +160,43 @@ attachHeader texts settings model _ attach =
|
|||||||
, title texts.openFileInNewTab
|
, title texts.openFileInNewTab
|
||||||
, class S.secondaryBasicButton
|
, class S.secondaryBasicButton
|
||||||
, class "ml-2"
|
, class "ml-2"
|
||||||
|
, classList [ ( "hidden", selectView ) ]
|
||||||
]
|
]
|
||||||
[ i [ class "fa fa-eye font-thin" ] []
|
[ i [ class "fa fa-eye font-thin" ] []
|
||||||
]
|
]
|
||||||
|
, a
|
||||||
|
[ classList
|
||||||
|
[ ( S.secondaryBasicButton ++ " text-sm", True )
|
||||||
|
, ( "bg-gray-200 dark:bg-bluegray-600", selectView )
|
||||||
|
, ( "hidden", not selectPossible )
|
||||||
|
, ( "ml-2", True )
|
||||||
|
]
|
||||||
|
, href "#"
|
||||||
|
, title selectToggleText
|
||||||
|
, onClick ToggleSelectView
|
||||||
|
]
|
||||||
|
[ i [ class "fa fa-tasks" ] []
|
||||||
|
]
|
||||||
|
, a
|
||||||
|
[ classList
|
||||||
|
[ ( S.deleteButton, True )
|
||||||
|
, ( "disabled", noAttachmentsSelected )
|
||||||
|
, ( "hidden", not selectPossible || not selectView )
|
||||||
|
, ( "ml-2", True )
|
||||||
|
]
|
||||||
|
, href "#"
|
||||||
|
, title texts.deleteAttachments
|
||||||
|
, onClick RequestDeleteSelected
|
||||||
|
]
|
||||||
|
[ i [ class "fa fa-trash" ] []
|
||||||
|
]
|
||||||
, MB.viewItem <|
|
, MB.viewItem <|
|
||||||
MB.Dropdown
|
MB.Dropdown
|
||||||
{ linkIcon = "fa fa-bars"
|
{ linkIcon = "fa fa-bars"
|
||||||
, linkClass =
|
, linkClass =
|
||||||
[ ( "ml-2", True )
|
[ ( "ml-2", True )
|
||||||
, ( S.secondaryBasicButton, True )
|
, ( S.secondaryBasicButton, True )
|
||||||
|
, ( "hidden", selectView )
|
||||||
]
|
]
|
||||||
, toggleMenu = ToggleAttachmentDropdown
|
, toggleMenu = ToggleAttachmentDropdown
|
||||||
, menuOpen = model.attachmentDropdownOpen
|
, menuOpen = model.attachmentDropdownOpen
|
||||||
@ -310,8 +355,33 @@ menuItem texts model pos attach =
|
|||||||
[ ( "bg-gray-300 dark:bg-bluegray-700 current-drop-target", enable )
|
[ ( "bg-gray-300 dark:bg-bluegray-700 current-drop-target", enable )
|
||||||
]
|
]
|
||||||
|
|
||||||
active =
|
iconClass =
|
||||||
model.visibleAttach == pos
|
case model.viewMode of
|
||||||
|
SelectView svm ->
|
||||||
|
if Set.member attach.id svm.ids then
|
||||||
|
"fa fa-check-circle ml-1"
|
||||||
|
|
||||||
|
else
|
||||||
|
"fa fa-circle ml-1"
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
"fa fa-check-circle ml-1"
|
||||||
|
|
||||||
|
visible =
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView _ ->
|
||||||
|
True
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
model.visibleAttach == pos
|
||||||
|
|
||||||
|
msg =
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView _ ->
|
||||||
|
ToggleAttachment attach.id
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
SetActiveAttachment pos
|
||||||
in
|
in
|
||||||
a
|
a
|
||||||
([ classList <|
|
([ classList <|
|
||||||
@ -322,18 +392,18 @@ menuItem texts model pos attach =
|
|||||||
, class "flex flex-col relative border rounded px-1 py-1 mr-2"
|
, class "flex flex-col relative border rounded px-1 py-1 mr-2"
|
||||||
, class " hover:shadow dark:hover:border-bluegray-500"
|
, class " hover:shadow dark:hover:border-bluegray-500"
|
||||||
, href "#"
|
, href "#"
|
||||||
, onClick (SetActiveAttachment pos)
|
, onClick msg
|
||||||
]
|
]
|
||||||
++ DD.draggable AttachDDMsg attach.id
|
++ DD.draggable AttachDDMsg attach.id
|
||||||
++ DD.droppable AttachDDMsg attach.id
|
++ DD.droppable AttachDDMsg attach.id
|
||||||
)
|
)
|
||||||
[ div
|
[ div
|
||||||
[ classList
|
[ classList
|
||||||
[ ( "hidden", not active )
|
[ ( "hidden", not visible )
|
||||||
]
|
]
|
||||||
, class "absolute right-1 top-1 text-blue-400 dark:text-lightblue-400 text-xl"
|
, class "absolute right-1 top-1 text-blue-400 dark:text-lightblue-400 text-xl"
|
||||||
]
|
]
|
||||||
[ i [ class "fa fa-check-circle ml-1" ] []
|
[ i [ class iconClass ] []
|
||||||
]
|
]
|
||||||
, div [ class "flex-grow" ]
|
, div [ class "flex-grow" ]
|
||||||
[ img
|
[ img
|
||||||
|
@ -23,21 +23,8 @@ import Comp.DetailEdit
|
|||||||
import Comp.Dropdown exposing (isDropdownChangeMsg)
|
import Comp.Dropdown exposing (isDropdownChangeMsg)
|
||||||
import Comp.Dropzone
|
import Comp.Dropzone
|
||||||
import Comp.EquipmentForm
|
import Comp.EquipmentForm
|
||||||
import Comp.ItemDetail.EditForm
|
|
||||||
import Comp.ItemDetail.FieldTabState as FTabState
|
import Comp.ItemDetail.FieldTabState as FTabState
|
||||||
import Comp.ItemDetail.Model
|
import Comp.ItemDetail.Model exposing (AttachmentRename, Model, Msg(..), NotesField(..), SaveNameState(..), SelectActionMode(..), UpdateResult, ViewMode(..), initSelectViewModel, isEditNotes, resultModel, resultModelCmd, resultModelCmdSub)
|
||||||
exposing
|
|
||||||
( AttachmentRename
|
|
||||||
, Model
|
|
||||||
, Msg(..)
|
|
||||||
, NotesField(..)
|
|
||||||
, SaveNameState(..)
|
|
||||||
, UpdateResult
|
|
||||||
, isEditNotes
|
|
||||||
, resultModel
|
|
||||||
, resultModelCmd
|
|
||||||
, resultModelCmdSub
|
|
||||||
)
|
|
||||||
import Comp.ItemMail
|
import Comp.ItemMail
|
||||||
import Comp.KeyInput
|
import Comp.KeyInput
|
||||||
import Comp.LinkTarget
|
import Comp.LinkTarget
|
||||||
@ -54,8 +41,6 @@ import Data.PersonUse
|
|||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
import DatePicker
|
import DatePicker
|
||||||
import Dict
|
import Dict
|
||||||
import Html exposing (..)
|
|
||||||
import Html.Attributes exposing (..)
|
|
||||||
import Html5.DragDrop as DD
|
import Html5.DragDrop as DD
|
||||||
import Http
|
import Http
|
||||||
import Page exposing (Page(..))
|
import Page exposing (Page(..))
|
||||||
@ -282,6 +267,23 @@ update key flags inav settings msg model =
|
|||||||
, attachRename = Nothing
|
, attachRename = Nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ToggleAttachment id ->
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView svm ->
|
||||||
|
let
|
||||||
|
svm_ =
|
||||||
|
if Set.member id svm.ids then
|
||||||
|
{ svm | ids = Set.remove id svm.ids }
|
||||||
|
|
||||||
|
else
|
||||||
|
{ svm | ids = Set.insert id svm.ids }
|
||||||
|
in
|
||||||
|
resultModel
|
||||||
|
{ model | viewMode = SelectView svm_ }
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
resultModel model
|
||||||
|
|
||||||
ToggleMenu ->
|
ToggleMenu ->
|
||||||
resultModel
|
resultModel
|
||||||
{ model | menuOpen = not model.menuOpen }
|
{ model | menuOpen = not model.menuOpen }
|
||||||
@ -938,6 +940,49 @@ update key flags inav settings msg model =
|
|||||||
in
|
in
|
||||||
resultModel model_
|
resultModel model_
|
||||||
|
|
||||||
|
RequestDeleteSelected ->
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView svm ->
|
||||||
|
if Set.isEmpty svm.ids then
|
||||||
|
resultModel model
|
||||||
|
|
||||||
|
else
|
||||||
|
let
|
||||||
|
confirmModal =
|
||||||
|
Comp.ConfirmModal.defaultSettings
|
||||||
|
DeleteSelectedConfirmed
|
||||||
|
AttachModalCancelled
|
||||||
|
"Ok"
|
||||||
|
"Cancel"
|
||||||
|
"Really delete these files?"
|
||||||
|
|
||||||
|
model_ =
|
||||||
|
{ model
|
||||||
|
| viewMode =
|
||||||
|
SelectView
|
||||||
|
{ svm
|
||||||
|
| action = DeleteSelected
|
||||||
|
}
|
||||||
|
, attachModal = Just confirmModal
|
||||||
|
}
|
||||||
|
in
|
||||||
|
resultModel model_
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
resultModel model
|
||||||
|
|
||||||
|
DeleteSelectedConfirmed ->
|
||||||
|
case model.viewMode of
|
||||||
|
SelectView svm ->
|
||||||
|
let
|
||||||
|
cmd =
|
||||||
|
Api.deleteAttachments flags svm.ids DeleteAttachResp
|
||||||
|
in
|
||||||
|
resultModelCmd ( { model | attachModal = Nothing, viewMode = SimpleView }, cmd )
|
||||||
|
|
||||||
|
SimpleView ->
|
||||||
|
resultModel model
|
||||||
|
|
||||||
AddFilesToggle ->
|
AddFilesToggle ->
|
||||||
resultModel
|
resultModel
|
||||||
{ model
|
{ model
|
||||||
@ -1360,7 +1405,11 @@ update key flags inav settings msg model =
|
|||||||
withSub ( model_, Cmd.none )
|
withSub ( model_, Cmd.none )
|
||||||
|
|
||||||
ToggleAttachMenu ->
|
ToggleAttachMenu ->
|
||||||
resultModel { model | attachMenuOpen = not model.attachMenuOpen }
|
resultModel
|
||||||
|
{ model
|
||||||
|
| attachMenuOpen = not model.attachMenuOpen
|
||||||
|
, viewMode = SimpleView
|
||||||
|
}
|
||||||
|
|
||||||
UiSettingsUpdated ->
|
UiSettingsUpdated ->
|
||||||
let
|
let
|
||||||
@ -1571,6 +1620,23 @@ update key flags inav settings msg model =
|
|||||||
in
|
in
|
||||||
resultModelCmd ( { model | itemModal = Nothing }, cmd )
|
resultModelCmd ( { model | itemModal = Nothing }, cmd )
|
||||||
|
|
||||||
|
ToggleSelectView ->
|
||||||
|
let
|
||||||
|
( nextView, cmd ) =
|
||||||
|
case model.viewMode of
|
||||||
|
SimpleView ->
|
||||||
|
( SelectView initSelectViewModel, Cmd.none )
|
||||||
|
|
||||||
|
SelectView _ ->
|
||||||
|
( SimpleView, Cmd.none )
|
||||||
|
in
|
||||||
|
withSub
|
||||||
|
( { model
|
||||||
|
| viewMode = nextView
|
||||||
|
}
|
||||||
|
, cmd
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- Helper
|
--- Helper
|
||||||
|
@ -15,6 +15,9 @@ type alias Texts =
|
|||||||
, viewExtractedData : String
|
, viewExtractedData : String
|
||||||
, reprocessFile : String
|
, reprocessFile : String
|
||||||
, deleteThisFile : String
|
, deleteThisFile : String
|
||||||
|
, selectModeTitle : String
|
||||||
|
, exitSelectMode : String
|
||||||
|
, deleteAttachments : String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -31,4 +34,7 @@ gb =
|
|||||||
, viewExtractedData = "View extracted data"
|
, viewExtractedData = "View extracted data"
|
||||||
, reprocessFile = "Re-process this file"
|
, reprocessFile = "Re-process this file"
|
||||||
, deleteThisFile = "Delete this file"
|
, deleteThisFile = "Delete this file"
|
||||||
|
, selectModeTitle = "Select Mode"
|
||||||
|
, exitSelectMode = "Exit Select Mode"
|
||||||
|
, deleteAttachments = "Delete attachments"
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user