Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
commonMain.jetbrains.datalore.plot.MonolithicCommon.kt Maven / Gradle / Ivy
/*
* Copyright (c) 2020. JetBrains s.r.o.
* Use of this source code is governed by the MIT license that can be found in the LICENSE file.
*/
package jetbrains.datalore.plot
import jetbrains.datalore.base.geometry.DoubleRectangle
import jetbrains.datalore.base.geometry.DoubleVector
import jetbrains.datalore.plot.builder.PlotContainerPortable
import jetbrains.datalore.plot.builder.assemble.PlotAssembler
import jetbrains.datalore.plot.builder.presentation.Defaults
import jetbrains.datalore.plot.config.BunchConfig
import jetbrains.datalore.plot.config.PlotConfig
import jetbrains.datalore.plot.config.PlotConfigClientSide
import jetbrains.datalore.plot.config.PlotConfigClientSideUtil
import jetbrains.datalore.plot.server.config.BackendSpecTransformUtil
import jetbrains.datalore.vis.svgToString.SvgToString
import kotlin.math.max
object MonolithicCommon {
/**
* Static SVG export
*/
fun buildSvgImagesFromRawSpecs(
plotSpec: MutableMap,
plotSize: DoubleVector?,
svgToString: SvgToString,
computationMessagesHandler: ((List) -> Unit)
): List {
@Suppress("NAME_SHADOWING")
val plotSpec = processRawSpecs(plotSpec, frontendOnly = false)
val buildResult = buildPlotsFromProcessedSpecs(plotSpec, plotSize)
if (buildResult.isError) {
val errorMessage = (buildResult as PlotsBuildResult.Error).error
throw RuntimeException(errorMessage)
}
val success = buildResult as PlotsBuildResult.Success
val computationMessages = success.buildInfos.flatMap { it.computationMessages }
if (computationMessages.isNotEmpty()) {
computationMessagesHandler(computationMessages)
}
return success.buildInfos.map {
val assembler = it.plotAssembler
val plot = assembler.createPlot()
val plotContainer = PlotContainerPortable(plot, it.size)
plotContainer.ensureContentBuilt()
plotContainer.svg
}.map { svgToString.render(it) }
}
fun buildPlotsFromProcessedSpecs(
plotSpec: MutableMap,
plotSize: DoubleVector?,
plotMaxWidth: Double? = null,
plotPreferredWidth: Double? = null
): PlotsBuildResult {
throwTestingErrors() // noop
@Suppress("NAME_SHADOWING")
val plotSize = plotSize?.let {
// Fix error (Batik):
// org.apache.batik.bridge.BridgeException: null:-1
// The attribute "height" of the element cannot be negative
DoubleVector(
max(0.0, it.x),
max(0.0, it.y)
)
}
PlotConfig.assertPlotSpecOrErrorMessage(plotSpec)
if (PlotConfig.isFailure(plotSpec)) {
val errorMessage = PlotConfig.getErrorMessage(plotSpec)
return PlotsBuildResult.Error(errorMessage)
}
return when {
PlotConfig.isPlotSpec(plotSpec) -> {
PlotsBuildResult.Success(
listOf(
buildSinglePlotFromProcessedSpecs(
plotSpec,
plotSize,
plotMaxWidth,
plotPreferredWidth
)
)
)
}
PlotConfig.isGGBunchSpec(plotSpec) -> buildGGBunchFromProcessedSpecs(
plotSpec,
plotMaxWidth,
plotPreferredWidth
)
else -> throw RuntimeException("Unexpected plot spec kind: " + PlotConfig.specKind(plotSpec))
}
}
private fun buildGGBunchFromProcessedSpecs(
bunchSpec: MutableMap,
maxWidth: Double?,
preferredWidth: Double?
): PlotsBuildResult {
val naturalSize = PlotSizeHelper.plotBunchSize(bunchSpec)
val scaledSize = preferredWidth?.let { w ->
naturalSize.mul(max(Defaults.MIN_PLOT_WIDTH, w) / naturalSize.x)
} ?: naturalSize
val neededSize = if (maxWidth != null && maxWidth < scaledSize.x) {
scaledSize.mul(max(Defaults.MIN_PLOT_WIDTH, maxWidth) / scaledSize.x)
} else {
scaledSize
}
val scalingCoef = neededSize.x / naturalSize.x
// val scalingCoef = maxWidth?.let {
// if (it < naturalBunchSize.x) {
// max(Defaults.MIN_PLOT_WIDTH, it) / naturalBunchSize.x
// } else {
// 1.0
// }
//
// } ?: 1.0
val bunchConfig = BunchConfig(bunchSpec)
if (bunchConfig.bunchItems.isEmpty()) return PlotsBuildResult.Error(
"No plots in the bunch"
)
val buildInfos = ArrayList()
for (bunchItem in bunchConfig.bunchItems) {
val plotSpec = bunchItem.featureSpec as MutableMap
val itemBuildInfoRaw = buildSinglePlotFromProcessedSpecs(
plotSpec,
PlotSizeHelper.bunchItemSize(bunchItem),
plotMaxWidth = null,
plotPreferredWidth = null
)
val itemBounds = DoubleRectangle(
DoubleVector(bunchItem.x, bunchItem.y).mul(scalingCoef),
itemBuildInfoRaw.size.mul(scalingCoef)
)
val itemBuildInfo = PlotBuildInfo(
itemBuildInfoRaw.plotAssembler,
itemBuildInfoRaw.processedPlotSpec,
itemBounds.origin,
itemBounds.dimension,
itemBuildInfoRaw.computationMessages
)
buildInfos.add(itemBuildInfo)
}
return PlotsBuildResult.Success(buildInfos)
}
private fun buildSinglePlotFromProcessedSpecs(
plotSpec: MutableMap,
plotSize: DoubleVector?,
plotMaxWidth: Double?,
plotPreferredWidth: Double?
): PlotBuildInfo {
val computationMessages = ArrayList()
val config = PlotConfigClientSide.create(plotSpec) {
computationMessages.addAll(it)
}
val preferredSize = PlotSizeHelper.singlePlotSize(
plotSpec,
plotSize,
plotMaxWidth,
plotPreferredWidth,
config.facets,
config.containsLiveMap
)
val assembler = createPlotAssembler(config)
return PlotBuildInfo(
assembler,
plotSpec,
DoubleVector.ZERO,
preferredSize,
computationMessages
)
}
private fun createPlotAssembler(
config: PlotConfigClientSide
): PlotAssembler {
return PlotConfigClientSideUtil.createPlotAssembler(config)
}
private fun throwTestingErrors() {
// testing errors
// throw RuntimeException()
// throw RuntimeException("My sudden crush")
// throw IllegalArgumentException("User configuration error")
// throw IllegalStateException("User configuration error")
// throw IllegalStateException() // Huh?
}
/**
* Applies all transformations to plot specifications.
* @param plotSpec: raw specifications of a single plot or GGBunch
*/
@Suppress("DuplicatedCode")
fun processRawSpecs(plotSpec: MutableMap, frontendOnly: Boolean): MutableMap {
PlotConfig.assertPlotSpecOrErrorMessage(plotSpec)
if (PlotConfig.isFailure(plotSpec)) {
return plotSpec
}
// Only "portable" transforms (not supported: raster image, any async transforms)
// Backend transforms
@Suppress("NAME_SHADOWING")
val plotSpec =
if (frontendOnly) {
plotSpec
} else {
BackendSpecTransformUtil.processTransform(plotSpec)
}
if (PlotConfig.isFailure(plotSpec)) {
return plotSpec
}
// Frontend transforms
return PlotConfigClientSide.processTransform(plotSpec)
}
sealed class PlotsBuildResult {
val isError: Boolean = this is Error
class Error(val error: String) : PlotsBuildResult()
class Success(
val buildInfos: List
) : PlotsBuildResult()
}
class PlotBuildInfo constructor(
val plotAssembler: PlotAssembler,
val processedPlotSpec: MutableMap,
val origin: DoubleVector,
val size: DoubleVector,
val computationMessages: List
) {
fun bounds(): DoubleRectangle {
return DoubleRectangle(origin, size)
}
}
}