2016-02-06 04:14:27 +00:00
|
|
|
#pragma once
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include "BaseControlDevice.h"
|
2017-11-20 04:08:23 +00:00
|
|
|
#include "KeyManager.h"
|
|
|
|
#include "IKeyManager.h"
|
|
|
|
#include "PPU.h"
|
2018-07-01 19:21:05 +00:00
|
|
|
#include "MemoryManager.h"
|
2016-02-06 04:14:27 +00:00
|
|
|
|
2017-11-20 04:08:23 +00:00
|
|
|
class Zapper : public BaseControlDevice
|
2016-06-22 23:23:08 +00:00
|
|
|
{
|
2017-11-20 04:08:23 +00:00
|
|
|
protected:
|
|
|
|
bool HasCoordinates() override { return true; }
|
2018-07-01 19:21:05 +00:00
|
|
|
|
2017-11-20 04:08:23 +00:00
|
|
|
string GetKeyNames() override
|
2016-06-22 23:23:08 +00:00
|
|
|
{
|
2017-11-20 04:08:23 +00:00
|
|
|
return "F";
|
2016-06-22 23:23:08 +00:00
|
|
|
}
|
|
|
|
|
2017-11-20 04:08:23 +00:00
|
|
|
enum Buttons { Fire };
|
2016-02-06 04:14:27 +00:00
|
|
|
|
2017-11-20 04:08:23 +00:00
|
|
|
void InternalSetStateFromInput() override
|
|
|
|
{
|
2018-07-14 02:19:26 +00:00
|
|
|
if(_console->GetSettings()->InputEnabled()) {
|
2017-11-20 04:08:23 +00:00
|
|
|
SetPressedState(Buttons::Fire, KeyManager::IsMouseButtonPressed(MouseButton::LeftButton));
|
|
|
|
}
|
|
|
|
|
|
|
|
MousePosition pos = KeyManager::GetMousePosition();
|
|
|
|
if(KeyManager::IsMouseButtonPressed(MouseButton::RightButton)) {
|
|
|
|
pos.X = -1;
|
|
|
|
pos.Y = -1;
|
|
|
|
}
|
|
|
|
SetCoordinates(pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsLightFound()
|
|
|
|
{
|
2018-07-01 19:21:05 +00:00
|
|
|
return StaticIsLightFound(GetCoordinates(), _console);
|
2017-11-20 04:08:23 +00:00
|
|
|
}
|
2016-02-06 04:14:27 +00:00
|
|
|
|
|
|
|
public:
|
2018-07-14 02:19:26 +00:00
|
|
|
Zapper(shared_ptr<Console> console, uint8_t port) : BaseControlDevice(console, port)
|
2017-11-20 04:08:23 +00:00
|
|
|
{
|
|
|
|
}
|
2016-02-06 04:14:27 +00:00
|
|
|
|
2017-11-20 04:08:23 +00:00
|
|
|
uint8_t ReadRAM(uint16_t addr) override
|
|
|
|
{
|
|
|
|
uint8_t output = 0;
|
2017-12-28 19:44:20 +00:00
|
|
|
if((IsExpansionDevice() && addr == 0x4017) || IsCurrentPort(addr)) {
|
2017-11-20 04:08:23 +00:00
|
|
|
output = (IsLightFound() ? 0 : 0x08) | (IsPressed(Zapper::Buttons::Fire) ? 0x10 : 0x00);
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
2016-02-06 04:14:27 +00:00
|
|
|
|
2017-11-20 04:08:23 +00:00
|
|
|
void WriteRAM(uint16_t addr, uint8_t value) override
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-07-01 19:21:05 +00:00
|
|
|
static bool StaticIsLightFound(MousePosition pos, shared_ptr<Console> console)
|
2017-11-20 04:08:23 +00:00
|
|
|
{
|
2018-07-01 19:21:05 +00:00
|
|
|
PPU* ppu = console ? console->GetPpu() : nullptr;
|
|
|
|
if(!ppu) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t scanline = ppu->GetCurrentScanline();
|
|
|
|
int32_t cycle = ppu->GetCurrentCycle();
|
2018-07-14 02:19:26 +00:00
|
|
|
int radius = (int)console->GetSettings()->GetZapperDetectionRadius();
|
2017-11-20 04:08:23 +00:00
|
|
|
|
|
|
|
if(pos.X >= 0 && pos.Y >= 0) {
|
|
|
|
for(int yOffset = -radius; yOffset <= radius; yOffset++) {
|
|
|
|
int yPos = pos.Y + yOffset;
|
|
|
|
if(yPos >= 0 && yPos < PPU::ScreenHeight) {
|
|
|
|
for(int xOffset = -radius; xOffset <= radius; xOffset++) {
|
|
|
|
int xPos = pos.X + xOffset;
|
|
|
|
if(xPos >= 0 && xPos < PPU::ScreenWidth) {
|
2018-07-01 19:21:05 +00:00
|
|
|
if(scanline >= yPos && (scanline - yPos <= 20) && (scanline != yPos || cycle > xPos) && ppu->GetPixelBrightness(xPos, yPos) >= 85) {
|
2017-11-20 04:08:23 +00:00
|
|
|
//Light cannot be detected if the Y/X position is further ahead than the PPU, or if the PPU drew a dark color
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2016-02-06 04:14:27 +00:00
|
|
|
};
|