/*****************/ /*** Layer RGB ***/ /*****************/ #include "stdafx.h" #include #include #include #include "GLEnabledView.h" #include #include #include "MapEdit.h" #include "MapEditDoc.h" #include "MapEditView.h" #include "MainFrm.h" #include "Core.h" #include "Layer.h" #include "LayerRGB.h" #include "Utils.h" #include "Select.h" #include "Export.h" #include "GUILayerRGB.h" #include "Elem.h" /*****************************************************************************/ char *CLayerRGB::RGBModeName[CLayerRGB::GUI_MODE_MAX]={"Paint","Lighten","Darken"}; float RGBAlpha=0.5f; int RateInc=5; #define MAX_UNDO 16 /*****************************************************************************/ /*****************************************************************************/ /*****************************************************************************/ // New Layer CLayerRGB::CLayerRGB(sLayerDef &Def) { InitLayer(Def); ShadeRGB.R=224; ShadeRGB.G=224; ShadeRGB.B=224; CurrentMode=0; CurrentBrush=0; CurrentRate=0; LastCursPos.x=-1; LastCursPos.y=-1; CurrentUndo=0; } /*****************************************************************************/ CLayerRGB::~CLayerRGB() { } /*****************************************************************************/ void CLayerRGB::InitLayer(sLayerDef &Def) { CLayer::InitLayer(Def); SetSize(Def.Width,Def.Height,true); BuildBrushList(); } /*****************************************************************************/ void CLayerRGB::Load(CFile *File,int Version) { InitLayer(LayerDef); File->Read(&ShadeRGB,sizeof(sRGBElem)); File->Read(&CurrentBrush,sizeof(int)); File->Read(&CurrentMode,sizeof(int)); File->Read(&SpareFlag,sizeof(bool)); // Read Map File->Read(&MapWidth,sizeof(int)); File->Read(&MapHeight,sizeof(int)); SetSize(MapWidth,MapHeight,false); for (int Y=0; YRead(&ThisElem,sizeof(sRGBElem)); } } } /*****************************************************************************/ void CLayerRGB::Save(CFile *File) { // Always Save current version File->Write(&ShadeRGB,sizeof(sRGBElem)); File->Write(&CurrentBrush,sizeof(int)); File->Write(&CurrentMode,sizeof(int)); File->Write(&SpareFlag,sizeof(bool)); // Read Map File->Write(&MapWidth,sizeof(int)); File->Write(&MapHeight,sizeof(int)); for (int Y=0; YWrite(&ThisElem,sizeof(sRGBElem)); } } } /*****************************************************************************/ void CLayerRGB::BuildBrushList() { GString Path; GString Filename; _finddata_t Find; long FileHandle; int Error=0; BrushList.clear(); GetExecPath(Path); Path+="\\Brushes\\"; Filename=Path+"*.bmp"; if( (FileHandle= _findfirst( Filename, &Find)) == -1L ) { ASSERT(!"No Brushes Found :o(\n"); return; } while (Error==0) { GString ThisFile=Path+Find.name; LoadBrush(ThisFile); Error=_findnext( FileHandle, &Find); } _findclose( FileHandle); } /*****************************************************************************/ void CLayerRGB::LoadBrush(const char *Name) { GFName Filename=Name; int BrushCount=BrushList.size(); BrushList.resize(BrushCount+1); sRGBBrush &NewBrush=BrushList[BrushCount]; Frame Frm; int W,H; NewBrush.Name=Filename.File(); Frm.LoadBMP(Name); W=Frm.GetWidth(); H=Frm.GetHeight(); NewBrush.W=W; NewBrush.H=H; NewBrush.XOfs=W/2; NewBrush.YOfs=H/2; NewBrush.Gfx.resize(W*H); int Ofs=0; for (int Y=0;YOffsetCam(CamPos,GetScaleFactor()); float ZoomW=Core->GetZoomW(); float ZoomH=Core->GetZoomH(); float ScrOfsX=(ZoomW/2); float ScrOfsY=(ZoomH/2); Vector3 &Scale=Core->GetScaleVector(); int StartX=(int)ThisCam.x; int StartY=(int)ThisCam.y; float ShiftX=ThisCam.x - (int)ThisCam.x; float ShiftY=ThisCam.y - (int)ThisCam.y; CLayerTile *ActionLayer=(CLayerTile*)Core->GetActionLayer(); if (StartX<0) StartX=0; if (StartY<0) StartY=0; int DrawW=ZoomW+8; int DrawH=ZoomH+8; if (StartX+DrawW>MapWidth) DrawW=MapWidth-StartX; if (StartY+DrawH>MapHeight) DrawH=MapHeight-StartY; glMatrixMode(GL_MODELVIEW); glPushMatrix(); glLoadIdentity(); glScalef(Scale.x,Scale.y,Scale.z); glTranslatef(-ShiftX,ShiftY,0); // Set scroll offset glTranslatef(-ScrOfsX,ScrOfsY,0); // Bring to top left corner for (int YLoop=0; YLoopGetMapElem(XPos,YPos); if (MapElem.Tile) { float fR=(1.0f/255.0f)*ThisElem.R; float fG=(1.0f/255.0f)*ThisElem.G; float fB=(1.0f/255.0f)*ThisElem.B; glLoadName (0); glBegin (GL_QUADS); glColor4f(fR,fG,fB,RGBAlpha); BuildGLQuad(0,1,0,1,0); glEnd(); } glTranslatef(1.0f,0,0); // Next X } glTranslatef(-DrawW,-1,0); // Next y, rewind to start X } glPopMatrix(); } /*****************************************************************************/ void CLayerRGB::RenderCursor(CCore *Core,Vector3 &CamPos,bool Is3d) { Vector3 ThisCam=Core->OffsetCam(CamPos,GetScaleFactor()); CPoint CursPos=Core->GetCursorPos(); if (CursPos.x<0 || CursPos.y<0) return; sRGBBrush &ThisBrush=BrushList[CurrentBrush]; CursPos.x-=ThisBrush.XOfs; CursPos.y-=ThisBrush.YOfs; CursPos.x-=(int)ThisCam.x; CursPos.y-=(int)ThisCam.y; float ZoomW=Core->GetZoomW(); float ZoomH=Core->GetZoomH(); float ScrOfsX=(ZoomW/2); float ScrOfsY=(ZoomH/2); Vector3 &Scale=Core->GetScaleVector(); float ShiftX=ThisCam.x - (int)ThisCam.x; float ShiftY=ThisCam.y - (int)ThisCam.y; std::vector &Gfx=ThisBrush.Gfx; int Ofs=0; int Rate=100-(CurrentRate*RateInc); float fRate=(1.0/100.f)*Rate; glMatrixMode(GL_MODELVIEW); glPushMatrix(); glLoadIdentity(); glScalef(Scale.x,Scale.y,Scale.z); glTranslatef(-ShiftX,ShiftY,0); // Set scroll offset glTranslatef(-ScrOfsX,ScrOfsY,0); // Bring to top left corner glTranslatef(CursPos.x,-CursPos.y,0); // Bring to top left corner for (int YLoop=0; YLoopGUIAdd(GUIRGB,IDD_LAYER_RGB); // Init BrushList ListSize=BrushList.size(); GUIRGB.m_BrushList.ResetContent(); for (i=0; iRedrawView(); } /*****************************************************************************/ void CLayerRGB::GUIKill(CCore *Core) { GUIChanged(Core); Core->GUIRemove(GUIRGB,IDD_LAYER_RGB); } /*****************************************************************************/ void CLayerRGB::GUIUpdate(CCore *Core) { GUIRGB.DisableCallback(); GUIRGB.m_ModeList.SetCurSel(CurrentMode); GUIRGB.m_BrushList.SetCurSel(CurrentBrush); GUIRGB.m_RateList.SetCurSel(CurrentRate); GUIRGB.SetRGB(ShadeRGB.R,ShadeRGB.G,ShadeRGB.B); GUIRGB.EnableCallback(); } /*****************************************************************************/ void CLayerRGB::GUIChanged(CCore *Core) { CurrentMode=GUIRGB.m_ModeList.GetCurSel(); CurrentBrush=GUIRGB.m_BrushList.GetCurSel(); CurrentRate=GUIRGB.m_RateList.GetCurSel(); GUIRGB.GetRGB(ShadeRGB.R,ShadeRGB.G,ShadeRGB.B); } /*****************************************************************************/ /*** Functions ***************************************************************/ /*****************************************************************************/ bool CLayerRGB::LButtonControl(CCore *Core,UINT nFlags, CPoint &CursorPos,bool DownFlag) { if (DownFlag) { if (CursorPos.x<0 || CursorPos.x>MapWidth) return(false); if (CursorPos.y<0 || CursorPos.y>MapHeight) return(false); CreateUndo(); Paint(Core,CursorPos); } else { LastCursPos.x=-1; LastCursPos.y=-1; } return(true); } /*****************************************************************************/ bool CLayerRGB::RButtonControl(CCore *Core,UINT nFlags, CPoint &CursorPos,bool DownFlag) { if (DownFlag) Grab(Core,CursorPos); return(true); } /*****************************************************************************/ bool CLayerRGB::MouseMove(CCore *Core,UINT nFlags, CPoint &CursorPos) { if (nFlags & MK_LBUTTON) { Paint(Core,CursorPos); } return(true); } /*****************************************************************************/ void CLayerRGB::CreateUndo() { sRGBUndo &ThisUndo=UndoList[CurrentUndo]; for (int y=0; y=MAX_UNDO) CurrentUndo=0; } /*****************************************************************************/ void CLayerRGB::Undo() { int Idx=CurrentUndo-1; if (Idx<0) Idx+=MAX_UNDO; sRGBUndo &ThisUndo=UndoList[Idx]; if (ThisUndo.Valid) { CurrentUndo=Idx; for (int y=0; y &Gfx=ThisBrush.Gfx; CPoint CursPos; int Ofs=0; CursPos.x=CursorPos.x-ThisBrush.XOfs; CursPos.y=CursorPos.y-ThisBrush.YOfs; for (int Y=0; Y=0 && Pos.x=0 && Pos.y255) RGB=255; Map[Pos.x][Pos.y].R=RGB; Map[Pos.x][Pos.y].G=RGB; Map[Pos.x][Pos.y].B=RGB; } } } } /*****************************************************************************/ void CLayerRGB::Grab(CCore *Core,CPoint &CursorPos) { /* if (CursorPos.x<0 || CursorPos.x>MapWidth) return; if (CursorPos.y<0 || CursorPos.y>MapHeight) return; CurrentRGB=Map[CursorPos.x][CursorPos.y]; GUIUpdate(Core); */ } /*****************************************************************************/ void CLayerRGB::BiFilter(CCore *Core) { for(int Y=0;Y>2 ) )>>1); GetFilterCol(Core,X-1,Y+0,SCol,SCount); GetFilterCol(Core,X+1,Y+0,SCol,SCount); GetFilterCol(Core,X+0,Y-1,SCol,SCount); GetFilterCol(Core,X+0,Y+1,SCol,SCount); if (SCount) { SCol/=SCount; SetFilterCol(Core,X,Y,SCol,2); } } } } /*****************************************************************************/ void CLayerRGB::TriFilter(CCore *Core) { for(int Y=0;Y>3 ) )>>1); GetFilterCol(Core,X-1,Y-1,SCol,SCount); GetFilterCol(Core,X+0,Y-1,SCol,SCount); GetFilterCol(Core,X+1,Y-1,SCol,SCount); GetFilterCol(Core,X-1,Y+0,SCol,SCount); GetFilterCol(Core,X+1,Y+0,SCol,SCount); GetFilterCol(Core,X-1,Y+1,SCol,SCount); GetFilterCol(Core,X+0,Y+1,SCol,SCount); GetFilterCol(Core,X+1,Y+1,SCol,SCount); if (SCount) { SCol/=SCount; SetFilterCol(Core,X,Y,SCol,2); } } } } /*****************************************************************************/ void CLayerRGB::STriFilter(CCore *Core) { for(int Y=0;Y>3 ) )>>1); // *(p-WIDTH-1)=c; // *(p-WIDTH+1)=c; // *(p+WIDTH-1)=c; // *(p+WIDTH+1)=c; GetFilterCol(Core,X-1,Y-1,SCol,SCount); GetFilterCol(Core,X+0,Y-1,SCol,SCount); GetFilterCol(Core,X+1,Y-1,SCol,SCount); GetFilterCol(Core,X-1,Y+0,SCol,SCount); GetFilterCol(Core,X+1,Y+0,SCol,SCount); GetFilterCol(Core,X-1,Y+1,SCol,SCount); GetFilterCol(Core,X+0,Y+1,SCol,SCount); GetFilterCol(Core,X+1,Y+1,SCol,SCount); if (SCount) { SCol/=SCount; SetFilterCol(Core,X-1,Y-1,SCol,1); SetFilterCol(Core,X+1,Y-1,SCol,1); SetFilterCol(Core,X-1,Y+1,SCol,1); SetFilterCol(Core,X+1,Y+1,SCol,1); } } } } /*****************************************************************************/ void CLayerRGB::GetFilterCol(CCore *Core,int X,int Y,int &SumCol,int &Count) { CLayerTile *ActionLayer=(CLayerTile*)Core->GetActionLayer(); if (X>=0 && X=0 && YGetMapElem(X,Y); if (MapElem.Tile) { sRGBElem &C=Map[X][Y]; SumCol+=C.R; Count++; } } } /*****************************************************************************/ void CLayerRGB::SetFilterCol(CCore *Core,int X,int Y,int Col,int Div) { CLayerTile *ActionLayer=(CLayerTile*)Core->GetActionLayer(); if (X>=0 && X=0 && YGetMapElem(X,Y); if (MapElem.Tile) { sRGBElem &ThisElem=Map[X][Y]; Col+=ThisElem.R; if (Div) Col/=Div; ThisElem.R=Col; ThisElem.G=Col; ThisElem.B=Col; } } } /*****************************************************************************/ void CLayerRGB::Export(CCore *Core,CExport &Exp) { sRGBCol RGB; Exp.ExportLayerHeader(LayerDef);//LAYER_TYPE_RGB,LayerDef.SubType,LayerDef.Width,LayerDef.Height); RGB.R=ShadeRGB.R; RGB.G=ShadeRGB.G; RGB.B=ShadeRGB.B; Exp.Write(&RGB,sizeof(sRGBCol)); for (int Y=0; Y