Introduce move based focusing

This commit is contained in:
Loic Guegan 2023-06-05 13:48:29 +02:00
parent 298714449a
commit 2a4a20beef
7 changed files with 100 additions and 12 deletions

View file

@ -10,19 +10,22 @@
* @brief CGEditor Window * @brief CGEditor Window
* *
*/ */
class MyFrame : public wxFrame, public cgeditor::CGEditor { class EditorPanel : public wxPanel, public cgeditor::CGEditor {
wxPaintDC *dc; wxPaintDC *dc;
/// @brief Used to test focus:
CMI::HalfMove *toFocus;
public: public:
MyFrame() EditorPanel(wxWindow *parent)
: wxFrame(NULL, wxID_ANY, "Hello World CGEditor"), CGEditor(), dc(NULL) { : wxPanel(parent), CGEditor(), dc(NULL) {
CreateStatusBar();
SetStatusText("CGEditor");
// Create a game // Create a game
CGEditor::status.Moves = BuildExampleGame(); CGEditor::status.Moves = BuildExampleGame();
// Set move to focus to the last move (downest move in the window!):
toFocus= CGEditor::status.Moves;
while(toFocus->GetMainline()!=nullptr){
toFocus=toFocus->GetMainline();
}
} }
private:
void OnExit(wxCommandEvent &event) { void OnExit(wxCommandEvent &event) {
(void)event; // Disable unused variable warning (void)event; // Disable unused variable warning
Close(true); } Close(true); }
@ -81,6 +84,19 @@ private:
} }
} }
void KeyboardEvent(wxKeyEvent& event){
switch ( event.GetKeyCode() )
{
case WXK_DOWN:
CGEditor::FocusOnMove(toFocus);
Refresh();
if(toFocus->GetParent()!=nullptr){
toFocus=toFocus->GetParent(); // Get previous move (not that it can be on the same line!)
}
break;
}
}
/** /**
* @brief Convenient fonction to center text * @brief Convenient fonction to center text
* *
@ -191,15 +207,30 @@ private:
DECLARE_EVENT_TABLE() DECLARE_EVENT_TABLE()
}; };
wxBEGIN_EVENT_TABLE(MyFrame, wxFrame) wxBEGIN_EVENT_TABLE(EditorPanel, wxPanel)
EVT_PAINT(MyFrame::OnPaint) EVT_PAINT(EditorPanel::OnPaint)
EVT_MOUSE_EVENTS(MyFrame::MouseEvent) EVT_MOUSE_EVENTS(EditorPanel::MouseEvent)
EVT_KEY_DOWN(EditorPanel::KeyboardEvent)
wxEND_EVENT_TABLE() wxEND_EVENT_TABLE()
class MainWindow: public wxFrame
{
wxPanel* editorPane;
public:
MainWindow(const wxString& title):wxFrame(NULL, wxID_ANY, title){
CreateStatusBar();
SetStatusText("CGEditor");
editorPane = new EditorPanel(this);
}
};
class MyApp : public wxApp { class MyApp : public wxApp {
public: public:
virtual bool OnInit() { virtual bool OnInit() {
MyFrame *frame = new MyFrame(); MainWindow *frame = new MainWindow("Hello World CGEditor");
frame->Show(true); frame->Show(true);
return true; return true;
} }

View file

@ -1,5 +1,5 @@
#include "CGEditor.hpp" #include "CGEditor.hpp"
#include <iostream>
namespace cgeditor { namespace cgeditor {
CGEditor::CGEditor() { CGEditor::CGEditor() {
@ -180,4 +180,10 @@ std::uint8_t CGEditor::GetNAGId(const std::string& symbol) const{
return 0; return 0;
} }
void CGEditor::FocusOnMove(CMI::HalfMove* move){
double X,Y;
MT->GetMoveXY(move,X,Y);
SBH->Focus(X+status.MoveWidth);
SBV->Focus(Y+status.MoveHeight);
}
} // namespace cgeditor } // namespace cgeditor

View file

@ -40,5 +40,7 @@ public:
std::string GetNAGSymbol(const std::uint8_t) const; std::string GetNAGSymbol(const std::uint8_t) const;
/// @brief Convert NAG symbol to id using the NagTable /// @brief Convert NAG symbol to id using the NagTable
std::uint8_t GetNAGId(const std::string&) const; std::uint8_t GetNAGId(const std::string&) const;
/// @brief Try to scroll to focus on a given move
void FocusOnMove(CMI::HalfMove*);
}; };
} // namespace cgeditor } // namespace cgeditor

View file

@ -69,6 +69,9 @@ std::uint32_t MoveTable::UpdateMoves(CMI::HalfMove *m, std::uint32_t line,
move_bound.text = m->GetSAN(); move_bound.text = m->GetSAN();
move_bound.ShouldApplyScroll = true; move_bound.ShouldApplyScroll = true;
bool isMouseOver = IsMouseOver(move_bound); bool isMouseOver = IsMouseOver(move_bound);
// Save in cache:
MovesStates[m].Bound=move_bound;
MovesStates[m].Line=line;
//---------- Update current focus move ---------- //---------- Update current focus move ----------
if (isMouseOver) { if (isMouseOver) {
@ -346,4 +349,9 @@ std::uint32_t MoveTable::DrawVariations(CMI::HalfMove *m, std::uint32_t line,
} }
return (line); return (line);
} }
void MoveTable::GetMoveXY(CMI::HalfMove* m, double &X, double &Y){
X=MovesStates.at(m).Bound.x;
Y=MovesStates.at(m).Bound.y;
}
} // namespace cgeditor } // namespace cgeditor

View file

@ -50,6 +50,8 @@ class MoveTable : public Component {
typedef struct MoveState { typedef struct MoveState {
bool IsFolded=false; bool IsFolded=false;
bool IsHidden=false; bool IsHidden=false;
int Line=0;
Element Bound;
} MoveState; } MoveState;
std::uint32_t UpdateMoves(CMI::HalfMove *, std::uint32_t, std::uint32_t,bool only_black); std::uint32_t UpdateMoves(CMI::HalfMove *, std::uint32_t, std::uint32_t,bool only_black);
std::int32_t CurrentMove; std::int32_t CurrentMove;
@ -66,5 +68,6 @@ public:
void Refresh(); void Refresh();
std::vector<Element> GetVariationsMarging() { return (VariationMargins); } std::vector<Element> GetVariationsMarging() { return (VariationMargins); }
void SyncCache(); void SyncCache();
void GetMoveXY(CMI::HalfMove* m, double &X, double &Y);
}; };
} // namespace cgeditor } // namespace cgeditor

View file

@ -109,4 +109,40 @@ void Scrollbar::Refresh() {
elements.push_back(bar); elements.push_back(bar);
} }
void Scrollbar::Focus(double XorY){
double MTCanvasHeight = status->CanvasHeight - status->ScrollbarWidth;
double MTCanvasWidth = status->CanvasWidth - status->ScrollbarWidth;
// Check horizontal or vertical and if should scroll
if(IsHorizontal && status->MoveTableMaxX > MTCanvasWidth){
// First normalize:
XorY-=MTCanvasWidth; // Do not scroll on first page
XorY=std::max(0.0,XorY); // Ensure focus is greater than 0
XorY=std::min(XorY,(status->MoveTableMaxX-MTCanvasWidth)); // Ensure focus is than scroll max
// Then compute percent and apply scroll
double percent = XorY / (status->MoveTableMaxX-MTCanvasWidth);
status->ScrollX =
-(status->MoveTableMaxX - MTCanvasWidth) * percent;
// Do not forget to update scrollbar:
double maxScroll = bg.width - bar.width;
bar.x = maxScroll * percent;
bar.x = std::max(bg.x, bar.x);
bar.x = std::min(bg.x + maxScroll, bar.x);
}
else if(status->MoveTableMaxY > MTCanvasHeight){
// First normalize:
XorY-=MTCanvasHeight; // Do not scroll on first page
XorY=std::max(0.0,XorY); // Ensure focus is greater than 0
XorY=std::min(XorY,(status->MoveTableMaxY-MTCanvasHeight)); // Ensure focus is than scroll max
// Then compute percent and apply scroll
double percent = XorY / (status->MoveTableMaxY-MTCanvasHeight);
status->ScrollY =
-(status->MoveTableMaxY - MTCanvasHeight) * percent;
// Do not forget to update scrollbar:
double maxScroll = bg.height - bar.height;
bar.y = maxScroll * percent;
bar.y = std::max(bg.y, bar.y);
bar.y = std::min(bg.y + maxScroll, bar.y);
}
}
} // namespace cgeditor } // namespace cgeditor

View file

@ -12,5 +12,7 @@ class Scrollbar : public Component {
public: public:
Scrollbar(Status* s,bool IsHorizontal); Scrollbar(Status* s,bool IsHorizontal);
void Refresh(); void Refresh();
/// @brief Goto a given graphical coordinate (if possible using current scroll range)
void Focus(double XorY);
}; };
} // namespace cgeditor } // namespace cgeditor