#ifndef INTERPRETER_LOCALOPCODES_H_INCLUDED
#define INTERPRETER_LOCALOPCODES_H_INCLUDED

#include "context.hpp"
#include "opcodes.hpp"
#include "runtime.hpp"

namespace Interpreter
{
    class OpStoreLocalShort : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer data = runtime[0].mInteger;
            int index = runtime[1].mInteger;

            runtime.getContext().setLocalShort(index, data);

            runtime.pop();
            runtime.pop();
        }
    };

    class OpStoreLocalLong : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer data = runtime[0].mInteger;
            int index = runtime[1].mInteger;

            runtime.getContext().setLocalLong(index, data);

            runtime.pop();
            runtime.pop();
        }
    };

    class OpStoreLocalFloat : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Float data = runtime[0].mFloat;
            int index = runtime[1].mInteger;

            runtime.getContext().setLocalFloat(index, data);

            runtime.pop();
            runtime.pop();
        }
    };

    class OpFetchIntLiteral : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer intValue = runtime.getIntegerLiteral(runtime[0].mInteger);
            runtime[0].mInteger = intValue;
        }
    };

    class OpFetchFloatLiteral : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Float floatValue = runtime.getFloatLiteral(runtime[0].mInteger);
            runtime[0].mFloat = floatValue;
        }
    };

    class OpFetchLocalShort : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            int index = runtime[0].mInteger;
            int value = runtime.getContext().getLocalShort(index);
            runtime[0].mInteger = value;
        }
    };

    class OpFetchLocalLong : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            int index = runtime[0].mInteger;
            int value = runtime.getContext().getLocalLong(index);
            runtime[0].mInteger = value;
        }
    };

    class OpFetchLocalFloat : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            int index = runtime[0].mInteger;
            float value = runtime.getContext().getLocalFloat(index);
            runtime[0].mFloat = value;
        }
    };

    class OpStoreGlobalShort : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer data = runtime[0].mInteger;
            int index = runtime[1].mInteger;

            std::string_view name = runtime.getStringLiteral(index);

            runtime.getContext().setGlobalShort(name, data);

            runtime.pop();
            runtime.pop();
        }
    };

    class OpStoreGlobalLong : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer data = runtime[0].mInteger;
            int index = runtime[1].mInteger;

            std::string_view name = runtime.getStringLiteral(index);

            runtime.getContext().setGlobalLong(name, data);

            runtime.pop();
            runtime.pop();
        }
    };

    class OpStoreGlobalFloat : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Float data = runtime[0].mFloat;
            int index = runtime[1].mInteger;

            std::string_view name = runtime.getStringLiteral(index);

            runtime.getContext().setGlobalFloat(name, data);

            runtime.pop();
            runtime.pop();
        }
    };

    class OpFetchGlobalShort : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            int index = runtime[0].mInteger;
            std::string_view name = runtime.getStringLiteral(index);
            Type_Integer value = runtime.getContext().getGlobalShort(name);
            runtime[0].mInteger = value;
        }
    };

    class OpFetchGlobalLong : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            int index = runtime[0].mInteger;
            std::string_view name = runtime.getStringLiteral(index);
            Type_Integer value = runtime.getContext().getGlobalLong(name);
            runtime[0].mInteger = value;
        }
    };

    class OpFetchGlobalFloat : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            int index = runtime[0].mInteger;
            std::string_view name = runtime.getStringLiteral(index);
            Type_Float value = runtime.getContext().getGlobalFloat(name);
            runtime[0].mFloat = value;
        }
    };

    template <bool TGlobal>
    class OpStoreMemberShort : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer data = runtime[0].mInteger;
            Type_Integer index = runtime[1].mInteger;
            ESM::RefId id = ESM::RefId::stringRefId(runtime.getStringLiteral(index));
            index = runtime[2].mInteger;
            std::string_view variable = runtime.getStringLiteral(index);

            runtime.getContext().setMemberShort(id, variable, data, TGlobal);

            runtime.pop();
            runtime.pop();
            runtime.pop();
        }
    };

    template <bool TGlobal>
    class OpStoreMemberLong : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer data = runtime[0].mInteger;
            Type_Integer index = runtime[1].mInteger;
            ESM::RefId id = ESM::RefId::stringRefId(runtime.getStringLiteral(index));
            index = runtime[2].mInteger;
            std::string_view variable = runtime.getStringLiteral(index);

            runtime.getContext().setMemberLong(id, variable, data, TGlobal);

            runtime.pop();
            runtime.pop();
            runtime.pop();
        }
    };

    template <bool TGlobal>
    class OpStoreMemberFloat : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Float data = runtime[0].mFloat;
            Type_Integer index = runtime[1].mInteger;
            ESM::RefId id = ESM::RefId::stringRefId(runtime.getStringLiteral(index));
            index = runtime[2].mInteger;
            std::string_view variable = runtime.getStringLiteral(index);

            runtime.getContext().setMemberFloat(id, variable, data, TGlobal);

            runtime.pop();
            runtime.pop();
            runtime.pop();
        }
    };

    template <bool TGlobal>
    class OpFetchMemberShort : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer index = runtime[0].mInteger;
            ESM::RefId id = ESM::RefId::stringRefId(runtime.getStringLiteral(index));
            index = runtime[1].mInteger;
            std::string_view variable = runtime.getStringLiteral(index);
            runtime.pop();

            int value = runtime.getContext().getMemberShort(id, variable, TGlobal);
            runtime[0].mInteger = value;
        }
    };

    template <bool TGlobal>
    class OpFetchMemberLong : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer index = runtime[0].mInteger;
            ESM::RefId id = ESM::RefId::stringRefId(runtime.getStringLiteral(index));
            index = runtime[1].mInteger;
            std::string_view variable = runtime.getStringLiteral(index);
            runtime.pop();

            int value = runtime.getContext().getMemberLong(id, variable, TGlobal);
            runtime[0].mInteger = value;
        }
    };

    template <bool TGlobal>
    class OpFetchMemberFloat : public Opcode0
    {
    public:
        void execute(Runtime& runtime) override
        {
            Type_Integer index = runtime[0].mInteger;
            ESM::RefId id = ESM::RefId::stringRefId(runtime.getStringLiteral(index));
            index = runtime[1].mInteger;
            std::string_view variable = runtime.getStringLiteral(index);
            runtime.pop();

            float value = runtime.getContext().getMemberFloat(id, variable, TGlobal);
            runtime[0].mFloat = value;
        }
    };
}

#endif