Jump to content
View in the app

A better way to browse. Learn more.

Forge Forums

A full-screen app on your home screen with push notifications, badges and more.

To install this app on iOS and iPadOS
  1. Tap the Share icon in Safari
  2. Scroll the menu and tap Add to Home Screen.
  3. Tap Add in the top-right corner.
To install this app on Android
  1. Tap the 3-dot menu (⋮) in the top-right corner of the browser.
  2. Tap Add to Home screen or Install app.
  3. Confirm by tapping Install.

Featured Replies

Posted

I am currently creating dozens (potentially 100+) different animations for the player's hand in first person view, and to make it easier on myself I have a system that lets me parse .json files for expressions, in a form like this:

 

{
	"transform":[
		{"translation":["__i * 0.56 - __swing_ratio * 1.75 + 0.9","-0.38", "-0.52"]},
		{"rotation":["0.0","90.0","0.0"]},
		{"rotation":["0.0","0.0","-90.0"]},
		{"rotation":["-25.0 + 25.0 * Math.cos(__swing_ratio * __swing_ratio * Math.PI)","0.0","0.0"]}
	]
}

 

these get read into a list of an HandTransformation class where type is the type of transformation.

 

public class HandTransformation
{
    public String type;
    public String[] transformation = new String[3];
}

 

At runtime, this method is called which evaluates the expression in the .json via a JS ScriptEngine, and does the appropriate transformations to the MatrixStacks object from the RenderHand event:

	matrixStack.clear();
        matrixStack.push();
        for (ItemTransformation itemTransformation:action.getTransformations())
        {
            String type = itemTransformation.type;
            Vector3f transform = getVector(itemTransformation, side, swingRatio, equippedRatio);

            if (type.equals("translation"))
            {
                matrixStack.translate(transform.getX(), transform.getY(), transform.getZ());
            }
            else if (type.equals("rotation"))
            {
                matrixStack.rotate(Vector3f.XP.rotationDegrees(transform.getX()));
                matrixStack.rotate(Vector3f.YP.rotationDegrees(transform.getY()));
                matrixStack.rotate(Vector3f.ZP.rotationDegrees(transform.getZ()));
            }
        }
        Minecraft.getInstance().getFirstPersonRenderer().renderItemSide(Minecraft.getInstance().player, itemStack, rightHanded ? ItemCameraTransforms.TransformType.FIRST_PERSON_RIGHT_HAND : ItemCameraTransforms.TransformType.FIRST_PERSON_LEFT_HAND, !rightHanded, matrixStack, buffer,light);
        matrixStack.pop();
    private Vector3f getVector(ItemTransformation itemTransformation, float side, float swingRatio, float equippedRatio)
    {
        String[] transformation = itemTransformation.transformation;
        String sideString = Float.toString(side);
        String swingRatioString = Float.toString(swingRatio);
        String equppedRatioString = Float.toString(equippedRatio);

        Vector3f transforms = new Vector3f();
        for (int i = 0; i < 3; i++)
        {
            String expression = transformation[i];
            expression.replaceAll("__i", sideString);
            expression.replaceAll("__swing_ratio", swingRatioString);
            expression.replaceAll("__equip_ratio", equppedRatioString);

            float result = evaluateScript(expression);

            switch(i)
            {
                case 0:
                    transforms.setX(result);
                    break;
                case 1:
                    transforms.setY(result);
                    break;
                default:
                    transforms.setZ(result);
            }
        }
        return transforms;
    }
    public float evaluateScript(String expression)
    {
        float result = 0.0f;
        ScriptEngineManager factory = new ScriptEngineManager();
        ScriptEngine engine = factory.getEngineByName("JavaScript");
        try
        {
            Object obj = engine.eval(expression);
            result = Float.parseFloat(obj.toString());
        }
        catch(ScriptException e)
        {
            e.printStackTrace();
        }
        return result;
    }

This all works, but as expected it cripples the framerate whenever it runs due to ScriptEngine. So, the next best thing is to apply the transformations to my own MatrixStacks object at initialization, for when I need them during gameplay.

 

Can I simply create my own MatrixStacks object at initialization (outside of the render event call) and not use the one provided by the event call? Or does it work more like the old GLStateManager and just wouldn't work?

 

Or better yet, is there an equally robust alternative to the JS script engine for evaluating mathematical expressions at runtime?

Edited by Turtledove

  • Author

Solved my own problem, I ended up using a parser rather than the ScriptEngine, which I should've done from the beginning.

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.
Note: Your post will require moderator approval before it will be visible.

Guest
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Reply to this topic...

Important Information

By using this site, you agree to our Terms of Use.

Configure browser push notifications

Chrome (Android)
  1. Tap the lock icon next to the address bar.
  2. Tap Permissions → Notifications.
  3. Adjust your preference.
Chrome (Desktop)
  1. Click the padlock icon in the address bar.
  2. Select Site settings.
  3. Find Notifications and adjust your preference.