package net.torvald.spriteassembler import com.badlogic.gdx.Gdx import com.badlogic.gdx.graphics.Pixmap import net.torvald.terrarum.AppLoader import net.torvald.terrarum.linearSearch import java.io.File /** * Assembles the single frame of the animation, outputs GDX Pixmap. * * The entire rendering is done by using pixmap. That is, no GPU access. * * Created by minjaesong on 2019-01-06. */ object AssembleSheetPixmap { operator fun invoke(properties: ADProperties): Pixmap { val canvas = Pixmap(properties.cols * properties.frameWidth, properties.rows * properties.frameHeight, Pixmap.Format.RGBA8888) canvas.blending = Pixmap.Blending.SourceOver // actually draw properties.transforms.forEach { t, _ -> drawThisFrame(t, canvas, properties) } return canvas } private fun drawThisFrame(frameName: String, canvas: Pixmap, properties: ADProperties ) { val theAnim = properties.getAnimByFrameName(frameName) val skeleton = theAnim.skeleton.joints.reversed() val transforms = properties.getTransform(frameName) val bodyparts = Array(skeleton.size) { // if file does not exist, null it val file = File("assets/" + properties.toFilename(skeleton[it].name)) //printdbg(this, "Loading file ${file.absolutePath}, exists: ${file.exists()}") /*return*/if (file.exists()) { Pixmap(Gdx.files.internal(file.path)) } else { null } } val transformList = AssembleFrameBase.makeTransformList(skeleton, transforms) val animRow = theAnim.row val animFrame = properties.getFrameNumberFromName(frameName) AppLoader.printdbg(this, "Frame to draw: $frameName (R$animRow C$animFrame)") drawFrame(animRow, animFrame, canvas, properties, bodyparts, transformList) bodyparts.forEach { it?.dispose() } } private fun drawFrame(row: Int, column: Int, canvas: Pixmap, props: ADProperties, bodyparts: Array, transformList: List> ) { val tmpFrame = Pixmap(props.frameWidth, props.frameHeight, Pixmap.Format.RGBA8888) bodyparts.forEachIndexed { index, image -> if (image != null) { val imgCentre = AssembleFrameBase.getCentreOf(image) val drawPos = transformList[index].second.invertY() + props.origin - imgCentre tmpFrame.drawPixmap(image, drawPos.x, drawPos.y) } } canvas.drawPixmap( tmpFrame, (column - 1) * props.frameWidth, (row - 1) * props.frameHeight ) tmpFrame.dispose() } } internal object AssembleFrameBase { /** * Returns joints list with tranform applied. * @param skeleton list of joints * @param transform ordered list of transforms should be applied. First come first serve. * @return List of pairs that contains joint name on left, final transform value on right */ fun makeTransformList(joints: List, transforms: List): List> { // make our mutable list val out = ArrayList>() joints.forEach { out.add(it.name to it.position) } // process transform queue transforms.forEach { transform -> if (transform.joint.name == ADProperties.ALL_JOINT_SELECT_KEY) { // transform applies to all joints for (c in 0 until out.size) { out[c] = out[c].first to (out[c].second + transform.translate) } } else { val i = out.linearSearch { it.first == transform.joint.name }!! // transform applies to one specific joint in the list (one specific joint is a search result) out[i] = out[i].first to (out[i].second + transform.translate) } } return out.toList() } fun getCentreOf(pixmap: Pixmap) = ADPropertyObject.Vector2i(pixmap.width / 2, pixmap.height / 2) }