Skip to content

Commit

Permalink
Make overview template more robust
Browse files Browse the repository at this point in the history
  • Loading branch information
Isti01 committed Oct 3, 2024
1 parent e27b9d3 commit 92cdc10
Show file tree
Hide file tree
Showing 11 changed files with 82 additions and 111 deletions.
37 changes: 27 additions & 10 deletions backend/src/main/kotlin/hu/bme/sch/cmsch/admin/GenerateOverview.kt
Original file line number Diff line number Diff line change
Expand Up @@ -41,16 +41,33 @@ fun GenerateOverview.sorter(): String {
}
}

fun GenerateOverview.extra(): String {
fun GenerateOverview.formatValue(value: Any?): Any =
if (renderer == OVERVIEW_TYPE_CDN_IMAGE) {
if (value is String && value.isNotBlank())
"/cdn/${cdnImageFolder}/${value}"
else
"" // empty image
} else if (renderer == OVERVIEW_TYPE_TEXT || renderer == OVERVIEW_TYPE_ICON) {
value ?: ""
} else {
value ?: 0
}

fun GenerateOverview.extra(): Array<Pair<String, Any>> {
return when (this.renderer) {
OVERVIEW_TYPE_ID -> ", \"width\":100, \"vertAlign\":\"middle\", \"visible\":false"
OVERVIEW_TYPE_TEXT -> ", \"vertAlign\":\"middle\""
OVERVIEW_TYPE_DATE -> ", \"vertAlign\":\"middle\",\"formatter\":\"datetime\""
OVERVIEW_TYPE_BOOLEAN -> ", \"formatter\":\"tickCross\", \"width\":120"
OVERVIEW_TYPE_CDN_IMAGE -> ", \"formatter\":\"image\", \"width\":120, \"formatterParams\":{\"height\":\"100px\"}"
OVERVIEW_TYPE_TIME -> ", \"vertAlign\":\"middle\""
OVERVIEW_TYPE_NUMBER -> ", \"vertAlign\":\"middle\""
OVERVIEW_TYPE_ICON -> ", \"vertAlign\":\"middle\", \"formatter\":\"enumIconsFormatter\", \"width\":120"
else -> ""
OVERVIEW_TYPE_ID -> arrayOf("width" to 100, "vertAlign" to "middle", "visible" to false)
OVERVIEW_TYPE_TEXT -> arrayOf("vertAlign" to "middle")
OVERVIEW_TYPE_DATE -> arrayOf("vertAlign" to "middle", "formatter" to "datetime")
OVERVIEW_TYPE_BOOLEAN -> arrayOf("formatter" to "tickCross", "width" to 120)
OVERVIEW_TYPE_CDN_IMAGE -> arrayOf(
"formatter" to "image",
"width" to 120,
"formatterParams" to mapOf("height" to "100px")
)

OVERVIEW_TYPE_TIME -> arrayOf("vertAlign" to "middle")
OVERVIEW_TYPE_NUMBER -> arrayOf("vertAlign" to "middle")
OVERVIEW_TYPE_ICON -> arrayOf("vertAlign" to "middle", "formatter" to "enumIconsFormatter", "width" to 120)
else -> emptyArray()
}
}
68 changes: 22 additions & 46 deletions backend/src/main/kotlin/hu/bme/sch/cmsch/admin/OverviewBuilder.kt
Original file line number Diff line number Diff line change
@@ -1,8 +1,5 @@
package hu.bme.sch.cmsch.admin

import com.fasterxml.jackson.core.type.TypeReference
import com.fasterxml.jackson.databind.ObjectMapper
import hu.bme.sch.cmsch.controller.admin.ControlAction
import java.io.ByteArrayOutputStream
import kotlin.reflect.KClass
import kotlin.reflect.KProperty1
Expand All @@ -15,65 +12,44 @@ class OverviewBuilder<T : Any>(val type: KClass<T>) {

private fun getColumnDefinitions(): List<Pair<KProperty1<T, *>, GenerateOverview>> {
return type.memberProperties.asSequence()
.filter { it.findAnnotation<GenerateOverview>() != null }
.map { Pair(it, it.findAnnotation<GenerateOverview>()!!) }
.filter { it.second.visible }
.sortedBy { it.second.order }
.toList()
.filter { it.findAnnotation<GenerateOverview>() != null }
.map { Pair(it, it.findAnnotation<GenerateOverview>()!!) }
.filter { it.second.visible }
.sortedBy { it.second.order }
.toList()
}

fun getColumnsAsJson(): String {
return "[" + (getColumnDefinitions().joinToString(",") {
"{\"title\":\"${it.second.columnName}\", " +
"\"field\":\"${it.first.name}\", " +
"\"hozAlign\":\"${it.second.alignment()}\", " +
"\"sorter\":\"${it.second.sorter()}\"" +
it.second.extra() +
"}"
}) + "]"
fun getColumns(): List<Map<String, Any>> = getColumnDefinitions().map {
mapOf(
"title" to it.second.columnName,
"field" to it.first.name,
"hozAlign" to it.second.alignment(),
"sorter" to it.second.sorter(),
*it.second.extra()
)
}

private fun formatValue(type: GenerateOverview, value: Any?): String {
return if (type.renderer == OVERVIEW_TYPE_CDN_IMAGE) {
if (value is String && !value.isNullOrBlank())
"\"/cdn/${type.cdnImageFolder}/${value}\""
else
"\"\"" // empty image
} else if (type.renderer == OVERVIEW_TYPE_TEXT || type.renderer == OVERVIEW_TYPE_ICON) {
"\"${(value ?: "").toString().replace("\"", "")}\""
} else {
(value ?: "0").toString().replace("\"", "")
}
}

fun getTableDataAsJson(overview: Iterable<T>): String {
fun getTableData(overview: Iterable<T>): List<Map<String, Any>> {
val fields = getColumnDefinitions()
return "[" + (overview.toList()
.map { row -> fields.map { "\"${it.first.name}\":${formatValue(it.second, it.first.get(row))}" } }
.joinToString(",") { row -> "{${row.joinToString(",")}}" }) + "]"
return overview.map { row ->
fields.map { it.first.name to it.second.formatValue(it.first.get(row)) }.toMap()
}
}

fun getInputs(): List<Pair<KProperty1<out Any, *>, GenerateInput>> {
return type.memberProperties.asSequence()
.filter { it.findAnnotation<GenerateInput>() != null }
.map { Pair(it, it.findAnnotation<GenerateInput>()!!) }
.filter { it.second.visible }
.sortedBy { it.second.order }
.toList()
.filter { it.findAnnotation<GenerateInput>() != null }
.map { Pair(it, it.findAnnotation<GenerateInput>()!!) }
.filter { it.second.visible }
.sortedBy { it.second.order }
.toList()
}

fun exportToCsv(entities: List<T>): String {
val outputStream = ByteArrayOutputStream()
csv.exportToCsv(entities, outputStream)
return outputStream.toString().trim()
}

fun toJson(list: List<ControlAction>, objectMapper: ObjectMapper): String {
return objectMapper
.writerFor(object : TypeReference<List<ControlAction>>() {})
.writeValueAsString(list)
}

}

object DetailsHelper {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -114,13 +114,11 @@ class MenuAdminController(
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", overviewDescriptor.getColumnsAsJson())
model.addAttribute("tableData", overviewDescriptor.getTableDataAsJson(fetchOverview()))
model.addAttribute("columnData", overviewDescriptor.getColumns())
model.addAttribute("tableData", overviewDescriptor.getTableData(fetchOverview()))

model.addAttribute("user", user)
model.addAttribute("controlActions", overviewDescriptor.toJson(
controlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActions)
model.addAttribute("buttonActions", buttonActions)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -120,16 +120,14 @@ class TaskAdminRateController(
model.addAttribute("titleSingular", titleSingular)
model.addAttribute("view", view)

model.addAttribute("columnData", descriptor.getColumnsAsJson())
model.addAttribute("columnData", descriptor.getColumns())
val submissions = transactionManager.transaction(readOnly = true) {
submittedRepository.findByTask_IdAndRejectedIsFalseAndApprovedIsFalseWithoutLobs(id)
}
model.addAttribute("tableData", descriptor.getTableDataAsJson(submissions))
model.addAttribute("tableData", descriptor.getTableData(submissions))

model.addAttribute("user", user)
model.addAttribute("controlActions", descriptor.toJson(
rateControlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", rateControlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", rateControlActions)
model.addAttribute("buttonActions", buttonActions.filter { it.permission.validate(user) })

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -90,13 +90,11 @@ class FilesByViewController(
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", overviewDescriptor.getColumnsAsJson())
model.addAttribute("tableData", overviewDescriptor.getTableDataAsJson(fetchOverview()))
model.addAttribute("columnData", overviewDescriptor.getColumns())
model.addAttribute("tableData", overviewDescriptor.getTableData(fetchOverview()))

model.addAttribute("user", user)
model.addAttribute("controlActions", overviewDescriptor.toJson(
controlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActions)
model.addAttribute("buttonActions", listOf<ButtonAction>())

Expand Down Expand Up @@ -126,8 +124,8 @@ class FilesByViewController(
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", submittedDescriptor.getColumnsAsJson())
model.addAttribute("tableData", submittedDescriptor.getTableDataAsJson(listFilesInView(id)))
model.addAttribute("columnData", submittedDescriptor.getColumns())
model.addAttribute("tableData", submittedDescriptor.getTableData(listFilesInView(id)))

model.addAttribute("user", user)
val controlActionForCategory = listOf(
Expand All @@ -152,9 +150,7 @@ class FilesByViewController(
basic = true
)
)
model.addAttribute("controlActions", overviewDescriptor.toJson(
controlActionForCategory.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActionForCategory.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActionForCategory)
model.addAttribute("buttonActions", listOf<ButtonAction>())

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -283,14 +283,12 @@ open class OneDeepEntityPage<T : IdentifiableEntity>(
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", descriptor.getColumnsAsJson())
model.addAttribute("columnData", descriptor.getColumns())
val overview = transactionManager.transaction(readOnly = true) { fetchOverview(user) }
model.addAttribute("tableData", descriptor.getTableDataAsJson(filterOverview(user, overview)))
model.addAttribute("tableData", descriptor.getTableData(filterOverview(user, overview)))

model.addAttribute("user", user)
model.addAttribute("controlActions", descriptor.toJson(
controlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActions)
model.addAttribute("buttonActions", buttonActions.filter { it.permission.validate(user) })
model.addAttribute("searchSettings", searchSettings)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -137,14 +137,12 @@ abstract class TwoDeepEntityPage<OUTER : IdentifiableEntity, INNER: Identifiable
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", outerDescriptor.getColumnsAsJson())
model.addAttribute("columnData", outerDescriptor.getColumns())
val overview = transactionManager.transaction(readOnly = true) { fetchOuterOverview() }
model.addAttribute("tableData", outerDescriptor.getTableDataAsJson(overview))
model.addAttribute("tableData", outerDescriptor.getTableData(overview))

model.addAttribute("user", user)
model.addAttribute("controlActions", outerDescriptor.toJson(
outerControlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", outerControlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", outerControlActions)
model.addAttribute("buttonActions", buttonActions.filter { it.permission.validate(user) })

Expand Down Expand Up @@ -174,14 +172,12 @@ abstract class TwoDeepEntityPage<OUTER : IdentifiableEntity, INNER: Identifiable
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", descriptor.getColumnsAsJson())
model.addAttribute("columnData", descriptor.getColumns())
val overview = transactionManager.transaction(readOnly = true) { filterOverview(user, fetchSublist(id)) }
model.addAttribute("tableData", descriptor.getTableDataAsJson(overview))
model.addAttribute("tableData", descriptor.getTableData(overview))

model.addAttribute("user", user)
model.addAttribute("controlActions", descriptor.toJson(
controlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActions)
model.addAttribute("buttonActions", buttonActions.filter { it.permission.validate(user) })

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -91,13 +91,11 @@ class AuditLogController(
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", descriptor.getColumnsAsJson())
model.addAttribute("tableData", descriptor.getTableDataAsJson(listFilesInView()))
model.addAttribute("columnData", descriptor.getColumns())
model.addAttribute("tableData", descriptor.getTableData(listFilesInView()))

model.addAttribute("user", user)
model.addAttribute("controlActions", descriptor.toJson(
controlActions.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActions.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActions)
model.addAttribute("buttonActions", listOf<ButtonAction>())

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -129,7 +129,7 @@ class PermissionGroupAssignPage(
model.addAttribute("description", "Válassz, hogy melyik felhasználó kapja meg a(z) ${permissionGroup.displayName} jogkört")
model.addAttribute("view", view)

model.addAttribute("columnData", descriptor.getColumnsAsJson())
model.addAttribute("columnData", descriptor.getColumns())
val overview = transactionManager.transaction(readOnly = true) {
userRepository.findAll()
.filter { permissionGroup.key.isNotEmpty() && !it.permissionGroups.split(",").contains(permissionGroup.key) }
Expand All @@ -143,12 +143,10 @@ class PermissionGroupAssignPage(
)
}
}
model.addAttribute("tableData", descriptor.getTableDataAsJson(filterOverview(user, overview)))
model.addAttribute("tableData", descriptor.getTableData(filterOverview(user, overview)))

model.addAttribute("user", user)
model.addAttribute("controlActions", descriptor.toJson(
controlActionsForView.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActionsForView.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActionsForView)
model.addAttribute("buttonActions", buttonActionsForView.filter { it.permission.validate(user) })
model.addAttribute("searchSettings", searchSettings)
Expand Down Expand Up @@ -188,4 +186,3 @@ class PermissionGroupAssignPage(
fun redirectToUsersView(@PathVariable id: Int) = "redirect:/admin/control/permission-groups-for-users/${id}"

}

Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,7 @@ class PermissionGroupUserListPage(
model.addAttribute("description", description)
model.addAttribute("view", view)

model.addAttribute("columnData", descriptor.getColumnsAsJson())
model.addAttribute("columnData", descriptor.getColumns())
val overview = transactionManager.transaction(readOnly = true) {
userRepository.findAllByPermissionGroupsNot("")
.filter { permissionGroup.key.isNotEmpty() && it.permissionGroups.split(",").contains(permissionGroup.key) }
Expand All @@ -154,12 +154,10 @@ class PermissionGroupUserListPage(
)
}
}
model.addAttribute("tableData", descriptor.getTableDataAsJson(filterOverview(user, overview)))
model.addAttribute("tableData", descriptor.getTableData(filterOverview(user, overview)))

model.addAttribute("user", user)
model.addAttribute("controlActions", descriptor.toJson(
controlActionsForView.filter { it.permission.validate(user) },
objectMapper))
model.addAttribute("controlActions", controlActionsForView.filter { it.permission.validate(user) })
model.addAttribute("allControlActions", controlActionsForView)
model.addAttribute("buttonActions", buttonActionsForView.filter { it.permission.validate(user) })
model.addAttribute("searchSettings", searchSettings)
Expand Down Expand Up @@ -201,4 +199,3 @@ class PermissionGroupUserListPage(
fun redirectToUsersView() = "redirect:/admin/control/permission-groups"

}

6 changes: 3 additions & 3 deletions backend/src/main/resources/templates/overview4.html
Original file line number Diff line number Diff line change
Expand Up @@ -110,8 +110,8 @@ <h5 th:text="${action.usageString}">Usage text</h5>
return dt.toFormat("yyyy-MM-dd HH:mm");
}

const buttons = JSON.parse(/*[[${controlActions}]]*/ "[]");
let tableData = JSON.parse(/*[[${tableData}]]*/ "[]");
const buttons = /*[[${controlActions}]]*/ [];
let tableData = /*[[${tableData}]]*/ [];
let view = /*[[${view}]]*/ "";
for (let rowId in tableData) {
const row = tableData[rowId];
Expand Down Expand Up @@ -140,7 +140,7 @@ <h5 th:text="${action.usageString}">Usage text</h5>
return enumElementMap[value] || "";
}

let columData = JSON.parse(/*[[${columnData}]]*/ "[]");
let columData = /*[[${columnData}]]*/ [];
for (let columnId in columData) {
if (columData[columnId].formatter === 'tickCross') {
columData[columnId]['formatterParams'] = {
Expand Down

0 comments on commit 92cdc10

Please sign in to comment.