604 lines
23 KiB
C++
604 lines
23 KiB
C++
/* FCE Ultra - NES/Famicom Emulator
|
|
*
|
|
* Copyright notice for this file:
|
|
* Copyright (C) 2002 Ben Parnell
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "common.h"
|
|
#include "ntview.h"
|
|
#include "debugger.h"
|
|
#include "..\..\fceu.h"
|
|
#include "..\..\debug.h"
|
|
#define INESPRIV
|
|
#include "..\..\cart.h"
|
|
#include "..\..\ines.h"
|
|
#include "..\..\palette.h"
|
|
#include "..\..\ppu.h"
|
|
|
|
HWND hNTView;
|
|
|
|
int NTViewPosX,NTViewPosY;
|
|
static uint8 palcache[32]; //palette cache //mbg merge 7/19/06 needed to be static
|
|
//uint8 ntcache0[0x400],ntcache1[0x400],ntcache2[0x400],ntcache3[0x400]; //cache CHR, fixes a refresh problem when right-clicking
|
|
uint8 *ntable0, *ntable1, *ntable2, *ntable3; //name table bitmap array
|
|
uint8 *ntablemirr[4];
|
|
uint8 ntcache[4][0x400]; //todo: use an array for the above variables too
|
|
int NTViewScanline=0,NTViewer=0;
|
|
int NTViewSkip,NTViewRefresh;
|
|
static int mouse_x,mouse_y; //todo: is static needed here? --mbg 7/19/06 - i think so
|
|
int redrawtables = 0;
|
|
int chrchanged = 0;
|
|
int ntmirroring, oldntmirroring = -1; //0 is horizontal, 1 is vertical, 2 is four screen.
|
|
int lockmirroring;
|
|
//int ntmirroring[4]; //these 4 ints are either 0, 1, 2, or 3 and tell where each of the 4 nametables point to
|
|
//uint8 *ntmirroringpointers[] = {&NTARAM[0x000],&NTARAM[0x400],ExtraNTARAM,ExtraNTARAM+0x400};
|
|
|
|
int horzscroll, vertscroll;
|
|
|
|
#define NTWIDTH 256
|
|
#define NTHEIGHT 240
|
|
//#define PATTERNBITWIDTH PATTERNWIDTH*3
|
|
#define NTDESTX 10
|
|
#define NTDESTY 15
|
|
#define ZOOM 1
|
|
|
|
//#define PALETTEWIDTH 32*4*4
|
|
//#define PALETTEHEIGHT 32*2
|
|
//#define PALETTEBITWIDTH PALETTEWIDTH*3
|
|
//#define PALETTEDESTX 10
|
|
//#define PALETTEDESTY 327
|
|
|
|
void UpdateMirroringButtons();
|
|
void ChangeMirroring();
|
|
|
|
static BITMAPINFO bmInfo; //todo is static needed here so it won't interefere with the pattern table viewer?
|
|
static HDC pDC,TmpDC0,TmpDC1;
|
|
static HBITMAP TmpBmp0,TmpBmp1;
|
|
static HGDIOBJ TmpObj0,TmpObj1;
|
|
|
|
static HDC TmpDC2,TmpDC3;
|
|
static HBITMAP TmpBmp2,TmpBmp3;
|
|
static HGDIOBJ TmpObj2,TmpObj3;
|
|
|
|
extern uint32 TempAddr, RefreshAddr;
|
|
extern uint8 XOffset;
|
|
|
|
int xpos, ypos;
|
|
int scrolllines = 1;
|
|
|
|
//if you pass this 1 then it will draw no matter what. If you pass it 0
|
|
//then it will draw if redrawtables is 1
|
|
void NTViewDoBlit(int autorefresh) {
|
|
if (!hNTView) return;
|
|
if (NTViewSkip < NTViewRefresh) {
|
|
NTViewSkip++;
|
|
return;
|
|
}
|
|
NTViewSkip=0;
|
|
|
|
if((redrawtables && !autorefresh) || (autorefresh) || (scrolllines)){
|
|
//todo: use switch and use bitblt for because its faster
|
|
switch(ntmirroring){
|
|
case 0: //horizontal
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
break;
|
|
case 1: //vertical
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
break;
|
|
case 2: //four screen
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC2,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC3,0,0,SRCCOPY);
|
|
break;
|
|
case 3: //single screen, table 0
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC0,0,0,SRCCOPY);
|
|
break;
|
|
case 4: //single screen, table 1
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC1,0,0,SRCCOPY);
|
|
break;
|
|
case 5: //single screen, table 2
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC2,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC2,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC2,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC2,0,0,SRCCOPY);
|
|
break;
|
|
case 6: //single screen, table 3
|
|
BitBlt(pDC,NTDESTX,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC3,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY,NTWIDTH,NTHEIGHT,TmpDC3,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC3,0,0,SRCCOPY);
|
|
BitBlt(pDC,NTDESTX+NTWIDTH,NTDESTY+NTHEIGHT,NTWIDTH,NTHEIGHT,TmpDC3,0,0,SRCCOPY);
|
|
break;
|
|
}
|
|
/*
|
|
if(ntmirroring == 0){ //horizontal
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
if(ntmirroring == 1){ //vertical
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
if(ntmirroring == 2){ //four screen
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC2,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC3,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
if(ntmirroring == 3){ //single screen, table 0
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
if(ntmirroring == 4){ //single screen, table 1
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
if(ntmirroring == 5){ //single screen, table 2
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC2,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC2,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC2,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC2,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
if(ntmirroring == 6){ //single screen, table 3
|
|
StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC3,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC3,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC3,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC3,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
*/
|
|
redrawtables = 0;
|
|
}
|
|
|
|
if(scrolllines){
|
|
SetROP2(pDC,R2_NOT);
|
|
|
|
//draw vertical line
|
|
MoveToEx(pDC,NTDESTX+xpos,NTDESTY,NULL);
|
|
LineTo(pDC,NTDESTX+xpos,NTDESTY+(NTHEIGHT*2));
|
|
|
|
//draw horizontal line
|
|
MoveToEx(pDC,NTDESTX,NTDESTY+ypos,NULL);
|
|
LineTo(pDC,NTDESTX+(NTWIDTH*2),NTDESTY+ypos);
|
|
|
|
SetROP2(pDC,R2_COPYPEN);
|
|
}
|
|
//StretchBlt(pDC,NTDESTX,NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC0,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
//StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY,NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC1,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
//StretchBlt(pDC,NTDESTX,NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC2,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
//StretchBlt(pDC,NTDESTX+(NTWIDTH*ZOOM),NTDESTY+(NTHEIGHT*ZOOM),NTWIDTH*ZOOM,NTHEIGHT*ZOOM,TmpDC3,0,NTHEIGHT-1,NTWIDTH,-NTHEIGHT,SRCCOPY);
|
|
}
|
|
|
|
void UpdateMirroringButtons(){
|
|
int i;
|
|
for(i = 0; i < 7;i++){
|
|
if(i != ntmirroring)CheckDlgButton(hNTView, i+1001, BST_UNCHECKED);
|
|
else CheckDlgButton(hNTView, i+1001, BST_CHECKED);
|
|
}
|
|
return;
|
|
}
|
|
|
|
void ChangeMirroring(){
|
|
switch(ntmirroring){
|
|
case 0:
|
|
vnapage[0] = vnapage[1] = &NTARAM[0x000];
|
|
vnapage[2] = vnapage[3] = &NTARAM[0x400];
|
|
break;
|
|
case 1:
|
|
vnapage[0] = vnapage[2] = &NTARAM[0x000];
|
|
vnapage[1] = vnapage[3] = &NTARAM[0x400];
|
|
break;
|
|
case 2:
|
|
vnapage[0] = &NTARAM[0x000];
|
|
vnapage[1] = &NTARAM[0x400];
|
|
vnapage[2] = ExtraNTARAM;
|
|
vnapage[3] = ExtraNTARAM+0x400;
|
|
break;
|
|
case 3:
|
|
vnapage[0] = vnapage[1] = vnapage[2] = vnapage[3] = &NTARAM[0x000];
|
|
break;
|
|
case 4:
|
|
vnapage[0] = vnapage[1] = vnapage[2] = vnapage[3] = &NTARAM[0x400];
|
|
break;
|
|
case 5:
|
|
vnapage[0] = vnapage[1] = vnapage[2] = vnapage[3] = ExtraNTARAM;
|
|
break;
|
|
case 6:
|
|
vnapage[0] = vnapage[1] = vnapage[2] = vnapage[3] = ExtraNTARAM+0x400;
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
|
|
INLINE void DrawChr(uint8 *pbitmap,uint8 *chr,int pal){
|
|
int y, x, tmp, index=0, p=0;
|
|
uint8 chr0, chr1;
|
|
//uint8 *table = &VPage[0][0]; //use the background table
|
|
//pbitmap += 3*
|
|
|
|
for (y = 0; y < 8; y++) { //todo: use index for y?
|
|
chr0 = chr[index];
|
|
chr1 = chr[index+8];
|
|
tmp=7;
|
|
for (x = 0; x < 8; x++) { //todo: use tmp for x?
|
|
p = (chr0>>tmp)&1;
|
|
p |= ((chr1>>tmp)&1)<<1;
|
|
p = palcache[p+(pal*4)];
|
|
tmp--;
|
|
|
|
*(uint8*)(pbitmap++) = palo[p].b;
|
|
*(uint8*)(pbitmap++) = palo[p].g;
|
|
*(uint8*)(pbitmap++) = palo[p].r;
|
|
}
|
|
index++;
|
|
pbitmap += (NTWIDTH*3)-24;
|
|
}
|
|
//index+=8;
|
|
//pbitmap -= (((PALETTEBITWIDTH>>2)<<3)-24);
|
|
}
|
|
|
|
void DrawNameTable(uint8 *bitmap, uint8 *table, uint8 *tablecache) {
|
|
int x,y,a, chr, ptable=0, cacheaddr0, cacheaddr1;//index=0;
|
|
uint8 *pbitmap = bitmap;
|
|
|
|
if(PPU[0]&0x10){ //use the correct pattern table based on this bit
|
|
ptable=0x1000;
|
|
}
|
|
|
|
pbitmap = bitmap;
|
|
for(y = 0;y < 30;y++){
|
|
for(x = 0;x < 32;x++){
|
|
cacheaddr0 = (y*32)+x;
|
|
cacheaddr1 = 0x3C0+((y>>2)<<3)+(x>>2);
|
|
if((tablecache == 0) || (table[cacheaddr0] != tablecache[cacheaddr0]) ||
|
|
(table[cacheaddr1] != tablecache[cacheaddr1])){
|
|
redrawtables = 1;
|
|
a = (table[0x3C0+((y>>2)<<3)+(x>>2)] & (3<<(((y&2)<<1)+(x&2)))) >> (((y&2)<<1)+(x&2));
|
|
//the commented out code below is all equilivent to the single line above.
|
|
//tmpx = x>>2;
|
|
//tmpy = y>>2;
|
|
//a = 0x3C0+(tmpy*8)+tmpx;
|
|
//if((((x>>1)&1) == 0) && (((y>>1)&1) == 0)) a = table[a]&0x3;
|
|
//if((((x>>1)&1) == 1) && (((y>>1)&1) == 0)) a = (table[a]&0xC)>>2;
|
|
//if((((x>>1)&1) == 0) && (((y>>1)&1) == 1)) a = (table[a]&0x30)>>4;
|
|
//if((((x>>1)&1) == 1) && (((y>>1)&1) == 1)) a = (table[a]&0xC0)>>6;
|
|
|
|
chr = table[y*32+x]*16;
|
|
DrawChr(pbitmap,&VPage[(ptable+chr)>>10][ptable+chr],a);
|
|
if(tablecache != 0){
|
|
tablecache[cacheaddr0] = table[cacheaddr0];
|
|
//tablecache[cacheaddr1] = table[cacheaddr1];
|
|
}
|
|
}
|
|
pbitmap += (8*3);
|
|
}
|
|
pbitmap += 7*((NTWIDTH*3));
|
|
}
|
|
|
|
if((tablecache != 0) && (redrawtables)){ //this copies the attribute tables to the cache if needed
|
|
memcpy(tablecache+0x3c0,table+0x3c0,0x40);
|
|
}
|
|
}
|
|
|
|
void FCEUD_UpdateNTView(int scanline, int drawall) {
|
|
if(!NTViewer) return;
|
|
if(scanline != -1 && scanline != NTViewScanline) return;
|
|
|
|
//uint8 *pbitmap = ppuv_palette;
|
|
if (!hNTView) return;
|
|
|
|
xpos = ((RefreshAddr & 0x400) >> 2) | ((RefreshAddr & 0x1F) << 3) | XOffset;
|
|
|
|
ypos = ((RefreshAddr & 0x3E0) >> 2) | ((RefreshAddr & 0x7000) >> 12);
|
|
if(RefreshAddr & 0x800) ypos += 240;
|
|
|
|
//char str[50];
|
|
//sprintf(str,"%d,%d,%d",horzscroll,vertscroll,ntmirroring);
|
|
//sprintf(str,"%08X %08X",TempAddr, RefreshAddr);
|
|
//SetDlgItemText(hNTView, 103, str);
|
|
|
|
if (NTViewSkip < NTViewRefresh) return;
|
|
|
|
if(chrchanged) drawall = 1;
|
|
|
|
//update palette only if required
|
|
if (memcmp(palcache,PALRAM,32) != 0) {
|
|
memcpy(palcache,PALRAM,32);
|
|
drawall = 1; //palette has changed, so redraw all
|
|
}
|
|
|
|
|
|
ntmirroring = -1;
|
|
if(vnapage[0] == vnapage[1])ntmirroring = 0;
|
|
if(vnapage[0] == vnapage[2])ntmirroring = 1;
|
|
if((vnapage[0] != vnapage[1]) && (vnapage[0] != vnapage[2]))ntmirroring = 2;
|
|
|
|
if((vnapage[0] == vnapage[1]) && (vnapage[1] == vnapage[2]) && (vnapage[2] == vnapage[3])){
|
|
if(vnapage[0] == &NTARAM[0x000])ntmirroring = 3;
|
|
if(vnapage[0] == &NTARAM[0x400])ntmirroring = 4;
|
|
if(vnapage[0] == ExtraNTARAM)ntmirroring = 5;
|
|
if(vnapage[0] == ExtraNTARAM+0x400)ntmirroring = 6;
|
|
}
|
|
|
|
if(oldntmirroring != ntmirroring){
|
|
UpdateMirroringButtons();
|
|
oldntmirroring = ntmirroring;
|
|
}
|
|
|
|
if(drawall){
|
|
DrawNameTable(ntable0,&NTARAM[0x000],0);
|
|
DrawNameTable(ntable1,&NTARAM[0x400],0);
|
|
DrawNameTable(ntable2,ExtraNTARAM,0);
|
|
DrawNameTable(ntable3,ExtraNTARAM+0x400,0);
|
|
} else {
|
|
if((ntmirroring == 0) || (ntmirroring == 1) || (ntmirroring == 3))
|
|
DrawNameTable(ntable0,&NTARAM[0x000],ntcache[0]);
|
|
if((ntmirroring == 0) || (ntmirroring == 1) || (ntmirroring == 4))
|
|
DrawNameTable(ntable1,&NTARAM[0x400],ntcache[1]);
|
|
if(ntmirroring == 2){
|
|
DrawNameTable(ntable2,ExtraNTARAM,ntcache[2]);
|
|
DrawNameTable(ntable3,ExtraNTARAM+0x400,ntcache[3]);
|
|
}
|
|
}
|
|
|
|
chrchanged = 0;
|
|
|
|
return;
|
|
}
|
|
|
|
void KillNTView() {
|
|
//GDI cleanup
|
|
DeleteObject(TmpBmp0);
|
|
SelectObject(TmpDC0,TmpObj0);
|
|
DeleteDC(TmpDC0);
|
|
DeleteObject(TmpBmp1);
|
|
SelectObject(TmpDC1,TmpObj1);
|
|
DeleteDC(TmpDC1);
|
|
DeleteObject(TmpBmp2);
|
|
SelectObject(TmpDC2,TmpObj2);
|
|
DeleteDC(TmpDC2);
|
|
DeleteObject(TmpBmp3);
|
|
SelectObject(TmpDC3,TmpObj3);
|
|
DeleteDC(TmpDC3);
|
|
ReleaseDC(hNTView,pDC);
|
|
|
|
DeleteObject (SelectObject (pDC, GetStockObject (BLACK_PEN))) ;
|
|
|
|
DestroyWindow(hNTView);
|
|
hNTView=NULL;
|
|
NTViewer=0;
|
|
NTViewSkip=0;
|
|
}
|
|
|
|
extern void StopSound(void);
|
|
|
|
BOOL CALLBACK NTViewCallB(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
|
|
RECT wrect;
|
|
char str[50];
|
|
int TileID, TileX, TileY, NameTable, PPUAddress;
|
|
|
|
switch(uMsg) {
|
|
case WM_INITDIALOG:
|
|
//SetWindowPos(hwndDlg,0,PPUViewPosX,PPUViewPosY,0,0,SWP_NOSIZE|SWP_NOZORDER|SWP_NOOWNERZORDER);
|
|
|
|
//prepare the bitmap attributes
|
|
//pattern tables
|
|
memset(&bmInfo.bmiHeader,0,sizeof(BITMAPINFOHEADER));
|
|
bmInfo.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
bmInfo.bmiHeader.biWidth = NTWIDTH;
|
|
bmInfo.bmiHeader.biHeight = -NTHEIGHT;
|
|
bmInfo.bmiHeader.biPlanes = 1;
|
|
bmInfo.bmiHeader.biBitCount = 24;
|
|
|
|
//create memory dcs
|
|
pDC = GetDC(hwndDlg); // GetDC(GetDlgItem(hwndDlg,101));
|
|
TmpDC0 = CreateCompatibleDC(pDC); //name table 0
|
|
TmpDC1 = CreateCompatibleDC(pDC); //name table 1
|
|
TmpDC2 = CreateCompatibleDC(pDC); //name table 2
|
|
TmpDC3 = CreateCompatibleDC(pDC); //name table 3
|
|
//TmpDC2 = CreateCompatibleDC(pDC); //palettes
|
|
|
|
//create bitmaps and select them into the memory dc's
|
|
TmpBmp0 = CreateDIBSection(pDC,&bmInfo,DIB_RGB_COLORS,(void**)&ntable0,0,0);
|
|
TmpObj0 = SelectObject(TmpDC0,TmpBmp0);
|
|
TmpBmp1 = CreateDIBSection(pDC,&bmInfo,DIB_RGB_COLORS,(void**)&ntable1,0,0);
|
|
TmpObj1 = SelectObject(TmpDC1,TmpBmp1);
|
|
TmpBmp2 = CreateDIBSection(pDC,&bmInfo,DIB_RGB_COLORS,(void**)&ntable2,0,0);
|
|
TmpObj2 = SelectObject(TmpDC2,TmpBmp2);
|
|
TmpBmp3 = CreateDIBSection(pDC,&bmInfo,DIB_RGB_COLORS,(void**)&ntable3,0,0);
|
|
TmpObj3 = SelectObject(TmpDC3,TmpBmp3);
|
|
|
|
//Refresh Trackbar
|
|
SendDlgItemMessage(hwndDlg,201,TBM_SETRANGE,0,(LPARAM)MAKELONG(0,25));
|
|
SendDlgItemMessage(hwndDlg,201,TBM_SETPOS,1,NTViewRefresh);
|
|
|
|
//Set Text Limit
|
|
SendDlgItemMessage(hwndDlg,102,EM_SETLIMITTEXT,3,0);
|
|
|
|
//force redraw the first time the PPU Viewer is opened
|
|
NTViewSkip=100;
|
|
|
|
SelectObject (pDC, CreatePen (PS_SOLID, 2, RGB (255, 255, 255))) ;
|
|
|
|
CheckDlgButton(hwndDlg, 1008, BST_CHECKED);
|
|
//clear cache
|
|
//memset(palcache,0,32);
|
|
//memset(ntcache0,0,0x400);
|
|
//memset(ntcache1,0,0x400);
|
|
//memset(ntcache2,0,0x400);
|
|
//memset(ntcache3,0,0x400);
|
|
|
|
NTViewer=1;
|
|
break;
|
|
case WM_PAINT:
|
|
NTViewDoBlit(1);
|
|
break;
|
|
case WM_CLOSE:
|
|
case WM_QUIT:
|
|
KillNTView();
|
|
break;
|
|
case WM_MOVING:
|
|
StopSound();
|
|
break;
|
|
case WM_MOVE:
|
|
GetWindowRect(hwndDlg,&wrect);
|
|
//NTViewPosX = wrect.left; //todo: is this needed?
|
|
//NTViewPosY = wrect.top;
|
|
break;
|
|
case WM_RBUTTONDBLCLK:
|
|
case WM_RBUTTONDOWN:
|
|
mouse_x = GET_X_LPARAM(lParam);
|
|
mouse_y = GET_Y_LPARAM(lParam);
|
|
/*if (((mouse_x >= PATTERNDESTX) && (mouse_x < (PATTERNDESTX+(PATTERNWIDTH*ZOOM)))) && (mouse_y >= PATTERNDESTY) && (mouse_y < (PATTERNDESTY+(PATTERNHEIGHT*ZOOM)))) {
|
|
if (pindex0 == 7) pindex0 = 0;
|
|
else pindex0++;
|
|
}
|
|
else if (((mouse_x >= PATTERNDESTX+(PATTERNWIDTH*ZOOM)+1) && (mouse_x < (PATTERNDESTX+(PATTERNWIDTH*ZOOM)*2+1))) && (mouse_y >= PATTERNDESTY) && (mouse_y < (PATTERNDESTY+(PATTERNHEIGHT*ZOOM)))) {
|
|
if (pindex1 == 7) pindex1 = 0;
|
|
else pindex1++;
|
|
}
|
|
UpdatePPUView(0);
|
|
PPUViewDoBlit();*/
|
|
break;
|
|
case WM_MOUSEMOVE:
|
|
mouse_x = GET_X_LPARAM(lParam);
|
|
mouse_y = GET_Y_LPARAM(lParam);
|
|
if((mouse_x > NTDESTX) && (mouse_x < NTDESTX+(NTWIDTH*2))
|
|
&& (mouse_y > NTDESTY) && (mouse_y < NTDESTY+(NTHEIGHT*2))){
|
|
TileX = (mouse_x-NTDESTX)/8;
|
|
TileY = (mouse_y-NTDESTY)/8;
|
|
sprintf(str,"X / Y: %0d / %0d",TileX,TileY);
|
|
SetDlgItemText(hwndDlg,104,str);
|
|
NameTable = (TileX/32)+((TileY/30)*2);
|
|
PPUAddress = 0x2000+(NameTable*0x400)+((TileY%30)*32)+(TileX%32);
|
|
sprintf(str,"PPU Address: %04X",PPUAddress);
|
|
SetDlgItemText(hwndDlg,105,str);
|
|
TileID = vnapage[(PPUAddress>>10)&0x3][PPUAddress&0x3FF];
|
|
sprintf(str,"Tile ID: %02X",TileID);
|
|
SetDlgItemText(hwndDlg,103,str);
|
|
}
|
|
|
|
/* if (((mouse_x >= PATTERNDESTX) && (mouse_x < (PATTERNDESTX+(PATTERNWIDTH*ZOOM)))) && (mouse_y >= PATTERNDESTY) && (mouse_y < (PATTERNDESTY+(PATTERNHEIGHT*ZOOM)))) {
|
|
mouse_x = (mouse_x-PATTERNDESTX)/(8*ZOOM);
|
|
mouse_y = (mouse_y-PATTERNDESTY)/(8*ZOOM);
|
|
sprintf(str,"Tile: $%X%X",mouse_y,mouse_x);
|
|
SetDlgItemText(hwndDlg,103,str);
|
|
SetDlgItemText(hwndDlg,104,"Tile:");
|
|
SetDlgItemText(hwndDlg,105,"Palettes");
|
|
}
|
|
else if (((mouse_x >= PATTERNDESTX+(PATTERNWIDTH*ZOOM)+1) && (mouse_x < (PATTERNDESTX+(PATTERNWIDTH*ZOOM)*2+1))) && (mouse_y >= PATTERNDESTY) && (mouse_y < (PATTERNDESTY+(PATTERNHEIGHT*ZOOM)))) {
|
|
mouse_x = (mouse_x-(PATTERNDESTX+(PATTERNWIDTH*ZOOM)+1))/(8*ZOOM);
|
|
mouse_y = (mouse_y-PATTERNDESTY)/(8*ZOOM);
|
|
sprintf(str,"Tile: $%X%X",mouse_y,mouse_x);
|
|
SetDlgItemText(hwndDlg,104,str);
|
|
SetDlgItemText(hwndDlg,103,"Tile:");
|
|
SetDlgItemText(hwndDlg,105,"Palettes");
|
|
}
|
|
else if (((mouse_x >= PALETTEDESTX) && (mouse_x < (PALETTEDESTX+PALETTEWIDTH))) && (mouse_y >= PALETTEDESTY) && (mouse_y < (PALETTEDESTY+PALETTEHEIGHT))) {
|
|
mouse_x = (mouse_x-PALETTEDESTX)/32;
|
|
mouse_y = (mouse_y-PALETTEDESTY)/32;
|
|
sprintf(str,"Palette: $%02X",palcache[(mouse_y<<4)|mouse_x]);
|
|
SetDlgItemText(hwndDlg,103,"Tile:");
|
|
SetDlgItemText(hwndDlg,104,"Tile:");
|
|
SetDlgItemText(hwndDlg,105,str);
|
|
}
|
|
else {
|
|
SetDlgItemText(hwndDlg,103,"Tile:");
|
|
SetDlgItemText(hwndDlg,104,"Tile:");
|
|
SetDlgItemText(hwndDlg,105,"Palettes");
|
|
}
|
|
*/
|
|
break;
|
|
case WM_NCACTIVATE:
|
|
sprintf(str,"%d",NTViewScanline);
|
|
SetDlgItemText(hwndDlg,102,str);
|
|
break;
|
|
case WM_COMMAND:
|
|
switch(HIWORD(wParam)) {
|
|
case EN_UPDATE:
|
|
GetDlgItemText(hwndDlg,102,str,4);
|
|
sscanf(str,"%d",&NTViewScanline);
|
|
if (NTViewScanline > 239) NTViewScanline = 239;
|
|
chrchanged = 1;
|
|
break;
|
|
case BN_CLICKED:
|
|
switch(LOWORD(wParam)) {
|
|
case 1001 :
|
|
case 1002 :
|
|
case 1003 :
|
|
case 1004 :
|
|
case 1005 :
|
|
case 1006 :
|
|
case 1007 :
|
|
oldntmirroring = ntmirroring = LOWORD(wParam)-1001;
|
|
ChangeMirroring();
|
|
break;
|
|
case 1008 :
|
|
scrolllines ^= 1;
|
|
chrchanged = 1;
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
case WM_HSCROLL:
|
|
if (lParam) { //refresh trackbar
|
|
NTViewRefresh = SendDlgItemMessage(hwndDlg,201,TBM_GETPOS,0,0);
|
|
}
|
|
break;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
void DoNTView() {
|
|
if (!GI) {
|
|
FCEUD_PrintError("You must have a game loaded before you can use the Name Table Viewer.");
|
|
return;
|
|
}
|
|
if (GI->type==GIT_NSF) {
|
|
FCEUD_PrintError("Sorry, you can't use the Name Table Viewer with NSFs.");
|
|
return;
|
|
}
|
|
|
|
if (!hNTView) hNTView = CreateDialog(fceu_hInstance,"NTVIEW",NULL,NTViewCallB);
|
|
if (hNTView) {
|
|
SetWindowPos(hNTView,HWND_TOP,0,0,0,0,SWP_NOSIZE|SWP_NOMOVE|SWP_NOOWNERZORDER);
|
|
FCEUD_UpdateNTView(-1,1);
|
|
NTViewDoBlit(1);
|
|
}
|
|
}
|
|
|