121 lines
2.7 KiB
C
121 lines
2.7 KiB
C
|
#pragma once
|
||
|
#include "stdafx.h"
|
||
|
#include "BaseControlDevice.h"
|
||
|
#include "KeyManager.h"
|
||
|
#include "Ppu.h"
|
||
|
#include "../Utilities/Serializer.h"
|
||
|
|
||
|
class SuperScope : public BaseControlDevice
|
||
|
{
|
||
|
private:
|
||
|
enum Buttons { Fire = 0, Cursor = 1, Turbo = 2, Pause = 3 };
|
||
|
uint32_t _stateBuffer = 0;
|
||
|
bool _prevFireButton = false;
|
||
|
bool _prevTurboButton = false;
|
||
|
bool _prevPauseButton = false;
|
||
|
bool _turbo = false;
|
||
|
Ppu *_ppu;
|
||
|
|
||
|
protected:
|
||
|
bool HasCoordinates() override { return true; }
|
||
|
|
||
|
string GetKeyNames() override
|
||
|
{
|
||
|
return "FCTP";
|
||
|
}
|
||
|
|
||
|
void InternalSetStateFromInput() override
|
||
|
{
|
||
|
SetPressedState(Buttons::Fire, KeyManager::IsMouseButtonPressed(MouseButton::LeftButton));
|
||
|
SetPressedState(Buttons::Cursor, KeyManager::IsMouseButtonPressed(MouseButton::RightButton));
|
||
|
SetPressedState(Buttons::Turbo, KeyManager::IsMouseButtonPressed(MouseButton::MiddleButton));
|
||
|
for(KeyMapping keyMapping : _keyMappings) {
|
||
|
SetPressedState(Buttons::Pause, KeyManager::IsKeyPressed(keyMapping.Start));
|
||
|
}
|
||
|
|
||
|
MousePosition pos = KeyManager::GetMousePosition();
|
||
|
SetCoordinates(pos);
|
||
|
}
|
||
|
|
||
|
void OnAfterSetState() override
|
||
|
{
|
||
|
MousePosition pos = GetCoordinates();
|
||
|
|
||
|
//Make the PPU latch the H/V counters at the mouse's position (offset slightly to make target in the center of the mouse cursor)
|
||
|
if(pos.X >= 0 && pos.Y >= 0) {
|
||
|
_ppu->SetLocationLatchRequest(pos.X + 10, std::max(0, pos.Y - 3));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Serialize(Serializer &s) override
|
||
|
{
|
||
|
BaseControlDevice::Serialize(s);
|
||
|
s.Stream(_stateBuffer, _prevTurboButton, _prevFireButton, _prevPauseButton, _turbo);
|
||
|
}
|
||
|
|
||
|
void RefreshStateBuffer() override
|
||
|
{
|
||
|
_stateBuffer = (uint32_t)ToByte();
|
||
|
_prevFireButton = IsPressed(Buttons::Fire);
|
||
|
_prevTurboButton = IsPressed(Buttons::Turbo);
|
||
|
_prevPauseButton = IsPressed(Buttons::Pause);
|
||
|
}
|
||
|
|
||
|
uint16_t ToByte()
|
||
|
{
|
||
|
uint16_t output = 0xFF00; //signature bits
|
||
|
|
||
|
if(!_prevTurboButton && IsPressed(Buttons::Turbo)) {
|
||
|
_turbo = !_turbo;
|
||
|
}
|
||
|
|
||
|
if((_turbo || !_prevFireButton) && IsPressed(Buttons::Fire)) {
|
||
|
output |= 0x01;
|
||
|
}
|
||
|
|
||
|
if(IsPressed(Buttons::Cursor)) {
|
||
|
output |= 0x02;
|
||
|
}
|
||
|
|
||
|
if(_turbo) {
|
||
|
output |= 0x04;
|
||
|
}
|
||
|
|
||
|
if(!_prevPauseButton && IsPressed(Buttons::Pause)) {
|
||
|
output |= 0x08;
|
||
|
}
|
||
|
|
||
|
if(GetCoordinates().X < 0 || GetCoordinates().Y < 0) {
|
||
|
output |= 0x40; //offscreen flag
|
||
|
}
|
||
|
|
||
|
return output;
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
SuperScope(Console* console, uint8_t port, KeyMappingSet keyMappings) : BaseControlDevice(console, port, keyMappings)
|
||
|
{
|
||
|
_ppu = console->GetPpu().get();
|
||
|
}
|
||
|
|
||
|
uint8_t ReadRam(uint16_t addr) override
|
||
|
{
|
||
|
uint8_t output = 0;
|
||
|
|
||
|
if(IsCurrentPort(addr)) {
|
||
|
StrobeProcessRead();
|
||
|
|
||
|
output = (_stateBuffer & 0x01);
|
||
|
|
||
|
_stateBuffer >>= 1;
|
||
|
_stateBuffer |= 0x8000;
|
||
|
}
|
||
|
|
||
|
return output;
|
||
|
}
|
||
|
|
||
|
void WriteRam(uint16_t addr, uint8_t value) override
|
||
|
{
|
||
|
StrobeProcessWrite(value);
|
||
|
}
|
||
|
};
|