diff options
Diffstat (limited to 'tools')
-rw-r--r-- | tools/mapedit/CMakeLists.txt | 1 | ||||
-rw-r--r-- | tools/mapedit/src/frame.cpp | 321 | ||||
-rw-r--r-- | tools/mapedit/src/frame.h | 32 | ||||
-rw-r--r-- | tools/mapedit/src/main.cpp | 2 | ||||
-rw-r--r-- | tools/mapedit/src/map.cpp | 289 | ||||
-rw-r--r-- | tools/mapedit/src/map.h | 45 | ||||
-rw-r--r-- | tools/mapedit/src/widget.cpp | 7 | ||||
-rw-r--r-- | tools/mapedit/src/widget.h | 3 | ||||
-rw-r--r-- | tools/mapedit/src/world.cpp | 369 | ||||
-rw-r--r-- | tools/mapedit/src/world.h | 45 |
10 files changed, 859 insertions, 255 deletions
diff --git a/tools/mapedit/CMakeLists.txt b/tools/mapedit/CMakeLists.txt index 9343f06..12af38c 100644 --- a/tools/mapedit/CMakeLists.txt +++ b/tools/mapedit/CMakeLists.txt | |||
@@ -44,6 +44,7 @@ add_executable(AromatherapyMapEditor | |||
44 | src/widget.cpp | 44 | src/widget.cpp |
45 | src/tile_widget.cpp | 45 | src/tile_widget.cpp |
46 | src/object.cpp | 46 | src/object.cpp |
47 | src/world.cpp | ||
47 | ) | 48 | ) |
48 | target_link_libraries(AromatherapyMapEditor ${ALL_LIBS}) | 49 | target_link_libraries(AromatherapyMapEditor ${ALL_LIBS}) |
49 | install(TARGETS AromatherapyMapEditor RUNTIME DESTINATION ${BIN_DIR}) | 50 | install(TARGETS AromatherapyMapEditor RUNTIME DESTINATION ${BIN_DIR}) |
diff --git a/tools/mapedit/src/frame.cpp b/tools/mapedit/src/frame.cpp index a38b861..9d489b8 100644 --- a/tools/mapedit/src/frame.cpp +++ b/tools/mapedit/src/frame.cpp | |||
@@ -5,6 +5,7 @@ | |||
5 | #include <wx/statline.h> | 5 | #include <wx/statline.h> |
6 | #include "panel.h" | 6 | #include "panel.h" |
7 | #include <list> | 7 | #include <list> |
8 | #include <exception> | ||
8 | 9 | ||
9 | static std::list<wxWindow*> openWindows; | 10 | static std::list<wxWindow*> openWindows; |
10 | 11 | ||
@@ -15,9 +16,18 @@ enum { | |||
15 | MENU_FILE_OPEN, | 16 | MENU_FILE_OPEN, |
16 | MENU_FILE_SAVE, | 17 | MENU_FILE_SAVE, |
17 | MENU_FILE_CLOSE, | 18 | MENU_FILE_CLOSE, |
19 | MENU_MAP_ADD_ROOT, | ||
20 | MENU_MAP_ADD_CHILD, | ||
18 | TOOL_FILE_NEW, | 21 | TOOL_FILE_NEW, |
19 | TOOL_FILE_OPEN, | 22 | TOOL_FILE_OPEN, |
20 | TOOL_FILE_SAVE | 23 | TOOL_FILE_SAVE, |
24 | TOOL_MAP_ADD_ROOT, | ||
25 | TOOL_MAP_ADD_CHILD, | ||
26 | MAP_EDITOR_NOTEBOOK, | ||
27 | MAP_EDITOR_TREE, | ||
28 | MAP_TITLE_TEXTBOX, | ||
29 | ADD_ENTITY_BUTTON, | ||
30 | CANCEL_ENTITY_BUTTON | ||
21 | }; | 31 | }; |
22 | 32 | ||
23 | wxBEGIN_EVENT_TABLE(MapeditFrame, wxFrame) | 33 | wxBEGIN_EVENT_TABLE(MapeditFrame, wxFrame) |
@@ -27,22 +37,41 @@ wxBEGIN_EVENT_TABLE(MapeditFrame, wxFrame) | |||
27 | EVT_MENU(MENU_FILE_NEW, MapeditFrame::OnNew) | 37 | EVT_MENU(MENU_FILE_NEW, MapeditFrame::OnNew) |
28 | EVT_MENU(MENU_FILE_OPEN, MapeditFrame::OnOpen) | 38 | EVT_MENU(MENU_FILE_OPEN, MapeditFrame::OnOpen) |
29 | EVT_MENU(MENU_FILE_SAVE, MapeditFrame::OnSave) | 39 | EVT_MENU(MENU_FILE_SAVE, MapeditFrame::OnSave) |
40 | EVT_MENU(MENU_FILE_CLOSE, MapeditFrame::OnClose) | ||
41 | EVT_MENU(MENU_MAP_ADD_ROOT, MapeditFrame::OnAddRoot) | ||
42 | EVT_MENU(MENU_MAP_ADD_CHILD, MapeditFrame::OnAddChild) | ||
30 | EVT_TOOL(TOOL_FILE_NEW, MapeditFrame::OnNew) | 43 | EVT_TOOL(TOOL_FILE_NEW, MapeditFrame::OnNew) |
31 | EVT_TOOL(TOOL_FILE_OPEN, MapeditFrame::OnOpen) | 44 | EVT_TOOL(TOOL_FILE_OPEN, MapeditFrame::OnOpen) |
32 | EVT_TOOL(TOOL_FILE_SAVE, MapeditFrame::OnSave) | 45 | EVT_TOOL(TOOL_FILE_SAVE, MapeditFrame::OnSave) |
33 | EVT_MENU(MENU_FILE_CLOSE, MapeditFrame::OnClose) | 46 | EVT_TOOL(TOOL_MAP_ADD_ROOT, MapeditFrame::OnAddRoot) |
47 | EVT_TOOL(TOOL_MAP_ADD_CHILD, MapeditFrame::OnAddChild) | ||
34 | EVT_CLOSE(MapeditFrame::OnExit) | 48 | EVT_CLOSE(MapeditFrame::OnExit) |
49 | EVT_NOTEBOOK_PAGE_CHANGED(MAP_EDITOR_NOTEBOOK, MapeditFrame::OnTabChange) | ||
50 | EVT_NOTEBOOK_PAGE_CHANGING(MAP_EDITOR_NOTEBOOK, MapeditFrame::OnTabChanging) | ||
51 | EVT_TREE_SEL_CHANGING(MAP_EDITOR_TREE, MapeditFrame::OnWillSelectMap) | ||
52 | EVT_TREE_SEL_CHANGED(MAP_EDITOR_TREE, MapeditFrame::OnDidSelectMap) | ||
53 | EVT_TREE_BEGIN_DRAG(MAP_EDITOR_TREE, MapeditFrame::OnWillDragMap) | ||
54 | EVT_TREE_END_DRAG(MAP_EDITOR_TREE, MapeditFrame::OnDidDragMap) | ||
55 | EVT_TEXT(MAP_TITLE_TEXTBOX, MapeditFrame::OnTitleChange) | ||
56 | EVT_BUTTON(ADD_ENTITY_BUTTON, MapeditFrame::OnAddEntity) | ||
57 | EVT_BUTTON(CANCEL_ENTITY_BUTTON, MapeditFrame::OnCancelAddEntity) | ||
35 | wxEND_EVENT_TABLE() | 58 | wxEND_EVENT_TABLE() |
36 | 59 | ||
37 | MapeditFrame::MapeditFrame(Map map, std::string filename) : wxFrame(NULL, wxID_ANY, "Map Editor", wxDefaultPosition, wxSize(GAME_WIDTH*3, GAME_HEIGHT*2)), map(map), filename(filename) | 60 | MapeditFrame::MapeditFrame(std::unique_ptr<World> world) : wxFrame(NULL, wxID_ANY, "Map Editor", wxDefaultPosition, wxSize(GAME_WIDTH*2+TILE_WIDTH*6*6+10+10+150, GAME_HEIGHT*3)) |
38 | { | 61 | { |
39 | this->map.frame = this; | 62 | this->world = std::move(world); |
63 | this->world->setParent(this); | ||
64 | currentMap = this->world->getLastMap(); | ||
40 | 65 | ||
41 | wxMenu* menuFile = new wxMenu; | 66 | menuFile = new wxMenu; |
42 | menuFile->Append(MENU_FILE_NEW, "New\tCtrl-N"); | 67 | menuFile->Append(MENU_FILE_NEW, "New\tCtrl-N"); |
43 | menuFile->Append(MENU_FILE_OPEN, "Open\tCtrl-O"); | 68 | menuFile->Append(MENU_FILE_OPEN, "Open\tCtrl-O"); |
44 | menuFile->Append(MENU_FILE_SAVE, "Save\tCtrl-S"); | 69 | menuFile->Append(MENU_FILE_SAVE, "Save\tCtrl-S"); |
45 | menuFile->Append(MENU_FILE_CLOSE, "Close\tCtrl-W"); | 70 | menuFile->Append(MENU_FILE_CLOSE, "Close\tCtrl-W"); |
71 | menuFile->AppendSeparator(); | ||
72 | menuFile->Append(MENU_MAP_ADD_ROOT, "New Map\tCtrl-Alt-N"); | ||
73 | menuFile->Append(MENU_MAP_ADD_CHILD, "New Child Map\tCtrl-Alt-Shift-N"); | ||
74 | menuFile->AppendSeparator(); | ||
46 | menuFile->Append(wxID_EXIT); | 75 | menuFile->Append(wxID_EXIT); |
47 | 76 | ||
48 | wxMenu* menuView = new wxMenu; | 77 | wxMenu* menuView = new wxMenu; |
@@ -59,22 +88,27 @@ MapeditFrame::MapeditFrame(Map map, std::string filename) : wxFrame(NULL, wxID_A | |||
59 | // Layout 2: Non-splitter between layout 3 and notebook | 88 | // Layout 2: Non-splitter between layout 3 and notebook |
60 | // Layout 3: Splitter between map editor and properties editor | 89 | // Layout 3: Splitter between map editor and properties editor |
61 | 90 | ||
62 | wxSplitterWindow* layout3 = new wxSplitterWindow(this, wxID_ANY); | 91 | wxSplitterWindow* layout1 = new wxSplitterWindow(this, wxID_ANY); |
92 | mapTree = new wxTreeCtrl(layout1, MAP_EDITOR_TREE, wxDefaultPosition, wxSize(200, 0), wxTR_HIDE_ROOT | wxTR_HAS_BUTTONS); | ||
93 | wxTreeItemId mapTreeRoot = mapTree->AddRoot("root"); | ||
94 | populateMapTree(mapTreeRoot, this->world->getRootMaps()); | ||
95 | |||
96 | wxPanel* layout2 = new wxPanel(layout1, wxID_ANY); | ||
97 | |||
98 | wxSplitterWindow* layout3 = new wxSplitterWindow(layout2, wxID_ANY); | ||
63 | layout3->SetSashGravity(1.0); | 99 | layout3->SetSashGravity(1.0); |
64 | layout3->SetMinimumPaneSize(20); | ||
65 | 100 | ||
66 | notebook = new wxNotebook(this, wxID_ANY); | 101 | notebook = new wxNotebook(layout2, MAP_EDITOR_NOTEBOOK); |
67 | 102 | ||
68 | tileEditor = new TileWidget(notebook, wxID_ANY, 6, 6, wxPoint(0,0), wxSize(TILE_WIDTH*6*6,TILE_HEIGHT*10*6)); | 103 | tileEditor = new TileWidget(notebook, wxID_ANY, 6, 6, wxPoint(0,0), wxSize(TILE_WIDTH*6*6,TILE_HEIGHT*10*6)); |
69 | notebook->AddPage(tileEditor, "Tile Chooser", true); | 104 | notebook->AddPage(tileEditor, "Tile Chooser", false); |
70 | 105 | ||
71 | mapEditor = new MapeditWidget(layout3, wxID_ANY, &this->map, tileEditor, wxPoint(0,0), wxSize(GAME_WIDTH*2, GAME_HEIGHT*2)); | 106 | mapEditor = new MapeditWidget(layout3, wxID_ANY, currentMap, tileEditor, wxPoint(0,0), wxSize(GAME_WIDTH*2, GAME_HEIGHT*2)); |
72 | mapEditor->frame = this; | 107 | mapEditor->frame = this; |
73 | 108 | ||
74 | // Set up property editor | 109 | // Set up property editor |
75 | wxPanel* propertyEditor = new wxPanel(layout3, wxID_ANY); | 110 | wxPanel* propertyEditor = new wxPanel(layout3, wxID_ANY); |
76 | titleBox = new wxTextCtrl(propertyEditor, wxID_ANY, map.getTitle()); | 111 | titleBox = new wxTextCtrl(propertyEditor, MAP_TITLE_TEXTBOX, currentMap->getTitle()); |
77 | titleBox->Bind(wxEVT_TEXT, &MapeditFrame::OnTitleChange, this); | ||
78 | 112 | ||
79 | wxStaticText* titleLabel = new wxStaticText(propertyEditor, wxID_ANY, "Title:"); | 113 | wxStaticText* titleLabel = new wxStaticText(propertyEditor, wxID_ANY, "Title:"); |
80 | 114 | ||
@@ -103,12 +137,9 @@ MapeditFrame::MapeditFrame(Map map, std::string filename) : wxFrame(NULL, wxID_A | |||
103 | entityTypeBox->Append(entry.second->getType(), entry.second.get()); | 137 | entityTypeBox->Append(entry.second->getType(), entry.second.get()); |
104 | } | 138 | } |
105 | 139 | ||
106 | addEntityButton = new wxButton(entityEditor, wxID_ANY, "Add Entity"); | 140 | addEntityButton = new wxButton(entityEditor, ADD_ENTITY_BUTTON, "Add Entity"); |
107 | addEntityButton->Bind(wxEVT_BUTTON, &MapeditFrame::OnAddEntity, this); | 141 | cancelEntityButton = new wxButton(entityEditor, CANCEL_ENTITY_BUTTON, "Cancel"); |
108 | |||
109 | cancelEntityButton = new wxButton(entityEditor, wxID_ANY, "Cancel"); | ||
110 | cancelEntityButton->Disable(); | 142 | cancelEntityButton->Disable(); |
111 | cancelEntityButton->Bind(wxEVT_BUTTON, &MapeditFrame::OnCancelAddEntity, this); | ||
112 | 143 | ||
113 | wxStaticText* entityInfoLabel = new wxStaticText(entityEditor, wxID_ANY, "Click and drag an entity to move it.\nRight click an entity to delete it."); | 144 | wxStaticText* entityInfoLabel = new wxStaticText(entityEditor, wxID_ANY, "Click and drag an entity to move it.\nRight click an entity to delete it."); |
114 | 145 | ||
@@ -129,51 +160,65 @@ MapeditFrame::MapeditFrame(Map map, std::string filename) : wxFrame(NULL, wxID_A | |||
129 | 160 | ||
130 | // Finish setting up the layouts | 161 | // Finish setting up the layouts |
131 | layout3->SplitHorizontally(mapEditor, propertyEditor); | 162 | layout3->SplitHorizontally(mapEditor, propertyEditor); |
163 | layout1->SplitVertically(mapTree, layout2); | ||
132 | 164 | ||
133 | notebook->Bind(wxEVT_NOTEBOOK_PAGE_CHANGED, &MapeditFrame::OnTabChange, this); | 165 | wxBoxSizer* sizer1 = new wxBoxSizer(wxHORIZONTAL); |
134 | notebook->Bind(wxEVT_NOTEBOOK_PAGE_CHANGING, &MapeditFrame::OnTabChanging, this); | 166 | sizer1->Add(layout1, 1, wxEXPAND, 0); |
167 | sizer1->Add(mapTree, 0, wxALIGN_TOP | wxALIGN_LEFT, 0); | ||
168 | sizer1->Add(layout2, 1, wxEXPAND, 0); | ||
169 | layout1->SetSizer(sizer1); | ||
170 | sizer1->SetSizeHints(layout1); | ||
135 | 171 | ||
136 | wxBoxSizer* sizer2 = new wxBoxSizer(wxHORIZONTAL); | 172 | wxBoxSizer* sizer2 = new wxBoxSizer(wxHORIZONTAL); |
137 | sizer2->Add(layout3, 1, wxEXPAND, 0); | 173 | sizer2->Add(layout3, 1, wxEXPAND, 0); |
138 | sizer2->Add(notebook, 0, wxALIGN_TOP | wxALIGN_CENTER_HORIZONTAL | wxLEFT | wxEXPAND, 2); | 174 | sizer2->Add(notebook, 0, wxALIGN_TOP | wxALIGN_CENTER_HORIZONTAL | wxLEFT | wxEXPAND, 2); |
139 | this->SetSizer(sizer2); | 175 | layout2->SetSizer(sizer2); |
140 | sizer2->SetSizeHints(this); | 176 | sizer2->SetSizeHints(layout2); |
141 | 177 | ||
142 | wxBoxSizer* splitterSizer = new wxBoxSizer(wxVERTICAL); | 178 | wxBoxSizer* splitterSizer = new wxBoxSizer(wxVERTICAL); |
143 | splitterSizer->Add(layout3, 1, wxEXPAND, 0); | 179 | splitterSizer->Add(layout3, 1, wxEXPAND, 0); |
144 | splitterSizer->Add(mapEditor, 1, wxEXPAND, 0); | 180 | splitterSizer->Add(mapEditor, 1, wxEXPAND, 0); |
145 | splitterSizer->Add(propertyEditor, 0, wxALIGN_TOP, wxALIGN_LEFT, 0); | 181 | splitterSizer->Add(propertyEditor, 0, wxALIGN_TOP | wxALIGN_LEFT, 0); |
146 | layout3->SetSizer(splitterSizer); | 182 | layout3->SetSizer(splitterSizer); |
147 | splitterSizer->SetSizeHints(layout3); | 183 | splitterSizer->SetSizeHints(layout3); |
148 | 184 | ||
149 | // Toolbar time! | 185 | // Toolbar time! |
150 | toolbar = CreateToolBar(); | 186 | toolbar = CreateToolBar(); |
151 | toolbar->AddTool(TOOL_FILE_NEW, "New", wxBitmap(wxImage("res/page_add.png"))); | 187 | toolbar->AddTool(TOOL_FILE_NEW, "New", wxBitmap(wxImage("res/page.png"))); |
152 | toolbar->AddTool(TOOL_FILE_OPEN, "Open", wxBitmap(wxImage("res/folder_page.png"))); | 188 | toolbar->AddTool(TOOL_FILE_OPEN, "Open", wxBitmap(wxImage("res/folder_page.png"))); |
153 | toolbar->AddTool(TOOL_FILE_SAVE, "Save", wxBitmap(wxImage("res/disk.png"))); | 189 | toolbar->AddTool(TOOL_FILE_SAVE, "Save", wxBitmap(wxImage("res/disk.png"))); |
154 | toolbar->EnableTool(TOOL_FILE_SAVE, this->map.getDirty()); | 190 | toolbar->AddSeparator(); |
191 | toolbar->AddTool(TOOL_MAP_ADD_ROOT, "Add Map", wxBitmap(wxImage("res/page_add.png"))); | ||
192 | toolbar->AddTool(TOOL_MAP_ADD_CHILD, "Add Child Map", wxBitmap(wxImage("res/page_white_add.png"))); | ||
193 | toolbar->EnableTool(TOOL_FILE_SAVE, this->world->getDirty()); | ||
155 | toolbar->Realize(); | 194 | toolbar->Realize(); |
195 | |||
196 | mapTree->SetFocusedItem(currentMap->getTreeItemId()); | ||
197 | SelectMap(currentMap); | ||
198 | |||
199 | Maximize(true); | ||
156 | } | 200 | } |
157 | 201 | ||
158 | void MapeditFrame::OnExit(wxCloseEvent& event) | 202 | void MapeditFrame::OnExit(wxCloseEvent& event) |
159 | { | 203 | { |
160 | if (event.CanVeto() && map.hasUnsavedChanges()) | 204 | if (event.CanVeto() && world->getDirty()) |
161 | { | 205 | { |
162 | switch (wxMessageBox("Current map has unsaved changes. Save before closing?", "Please confirm", wxICON_QUESTION|wxYES_NO|wxCANCEL, this)) | 206 | switch (wxMessageBox("One or more maps have unsaved changes. Save before closing?", "Please confirm", wxICON_QUESTION|wxYES_NO|wxCANCEL, this)) |
163 | { | 207 | { |
164 | case wxYES: | 208 | case wxYES: |
165 | if (filename == "") | 209 | if (world->getFilename() == "") |
166 | { | 210 | { |
167 | wxFileDialog saveFileDialog(this, "Save map", "", "", "XML files (*.xml)|*.xml", wxFD_SAVE); | 211 | wxFileDialog saveFileDialog(this, "Save world", "", "", "XML files (*.xml)|*.xml", wxFD_SAVE); |
168 | if (saveFileDialog.ShowModal() == wxID_CANCEL) | 212 | if (saveFileDialog.ShowModal() == wxID_CANCEL) |
169 | { | 213 | { |
170 | return; | 214 | return; |
171 | } | 215 | } |
172 | 216 | ||
173 | filename = saveFileDialog.GetPath().ToStdString(); | 217 | world->save(saveFileDialog.GetPath().ToStdString(), mapTree); |
218 | } else { | ||
219 | world->save(world->getFilename(), mapTree); | ||
174 | } | 220 | } |
175 | 221 | ||
176 | map.save(filename); | ||
177 | break; | 222 | break; |
178 | 223 | ||
179 | case wxCANCEL: | 224 | case wxCANCEL: |
@@ -204,34 +249,34 @@ void MapeditFrame::ZoomOut(wxCommandEvent&) | |||
204 | 249 | ||
205 | void MapeditFrame::OnNew(wxCommandEvent&) | 250 | void MapeditFrame::OnNew(wxCommandEvent&) |
206 | { | 251 | { |
207 | NewMap(); | 252 | NewWorld(); |
208 | } | 253 | } |
209 | 254 | ||
210 | void MapeditFrame::OnOpen(wxCommandEvent&) | 255 | void MapeditFrame::OnOpen(wxCommandEvent&) |
211 | { | 256 | { |
212 | wxFileDialog openFileDialog(this, "Open map", "", "", "XML files (*.xml)|*.xml", wxFD_OPEN|wxFD_FILE_MUST_EXIST); | 257 | wxFileDialog openFileDialog(this, "Open world", "", "", "XML files (*.xml)|*.xml", wxFD_OPEN|wxFD_FILE_MUST_EXIST); |
213 | if (openFileDialog.ShowModal() == wxID_CANCEL) | 258 | if (openFileDialog.ShowModal() == wxID_CANCEL) |
214 | { | 259 | { |
215 | return; | 260 | return; |
216 | } | 261 | } |
217 | 262 | ||
218 | OpenMap(openFileDialog.GetPath().c_str()); | 263 | OpenWorld(openFileDialog.GetPath().ToStdString()); |
219 | } | 264 | } |
220 | 265 | ||
221 | void MapeditFrame::OnSave(wxCommandEvent&) | 266 | void MapeditFrame::OnSave(wxCommandEvent&) |
222 | { | 267 | { |
223 | if (filename == "") | 268 | if (world->getFilename() == "") |
224 | { | 269 | { |
225 | wxFileDialog saveFileDialog(this, "Save map", "", "", "XML files (*.xml)|*.xml", wxFD_SAVE); | 270 | wxFileDialog saveFileDialog(this, "Save world", "", "", "XML files (*.xml)|*.xml", wxFD_SAVE); |
226 | if (saveFileDialog.ShowModal() == wxID_CANCEL) | 271 | if (saveFileDialog.ShowModal() == wxID_CANCEL) |
227 | { | 272 | { |
228 | return; | 273 | return; |
229 | } | 274 | } |
230 | 275 | ||
231 | filename = saveFileDialog.GetPath().ToStdString(); | 276 | world->save(saveFileDialog.GetPath().ToStdString(), mapTree); |
277 | } else { | ||
278 | world->save(world->getFilename(), mapTree); | ||
232 | } | 279 | } |
233 | |||
234 | map.save(filename); | ||
235 | } | 280 | } |
236 | 281 | ||
237 | void MapeditFrame::OnClose(wxCommandEvent&) | 282 | void MapeditFrame::OnClose(wxCommandEvent&) |
@@ -252,24 +297,8 @@ void MapeditFrame::OnQuit(wxCommandEvent&) | |||
252 | 297 | ||
253 | void MapeditFrame::OnTitleChange(wxCommandEvent&) | 298 | void MapeditFrame::OnTitleChange(wxCommandEvent&) |
254 | { | 299 | { |
255 | map.setTitle(titleBox->GetLineText(0).ToStdString()); | 300 | currentMap->setTitle(titleBox->GetValue().ToStdString()); |
256 | } | 301 | mapTree->SetItemText(currentMap->getTreeItemId(), currentMap->getTitle()); |
257 | |||
258 | void MapeditFrame::NewMap() | ||
259 | { | ||
260 | LaunchWindow(Map(), ""); | ||
261 | } | ||
262 | |||
263 | void MapeditFrame::OpenMap(const char* filename) | ||
264 | { | ||
265 | LaunchWindow(Map(filename), filename); | ||
266 | } | ||
267 | |||
268 | void MapeditFrame::LaunchWindow(Map map, const char* filename) | ||
269 | { | ||
270 | MapeditFrame* frame = new MapeditFrame(map, filename); | ||
271 | frame->closer = openWindows.insert(end(openWindows), frame); | ||
272 | frame->Show(true); | ||
273 | } | 302 | } |
274 | 303 | ||
275 | void MapeditFrame::OnTabChange(wxBookCtrlEvent& event) | 304 | void MapeditFrame::OnTabChange(wxBookCtrlEvent& event) |
@@ -317,11 +346,113 @@ void MapeditFrame::OnCancelAddEntity(wxCommandEvent&) | |||
317 | mapEditor->CancelAddingEntity(); | 346 | mapEditor->CancelAddingEntity(); |
318 | } | 347 | } |
319 | 348 | ||
349 | void MapeditFrame::OnAddRoot(wxCommandEvent&) | ||
350 | { | ||
351 | auto map = world->newMap(); | ||
352 | wxTreeItemId node = mapTree->AppendItem(mapTree->GetItemParent(mapTree->GetSelection()), map->getTitle()); | ||
353 | map->setTreeItemId(node); | ||
354 | mapTree->SetItemData(node, new MapPtrCtr(map.get())); | ||
355 | mapTree->SetFocusedItem(node); | ||
356 | SelectMap(map.get()); | ||
357 | } | ||
358 | |||
359 | void MapeditFrame::OnAddChild(wxCommandEvent&) | ||
360 | { | ||
361 | auto map = world->newMap(); | ||
362 | wxTreeItemId node = mapTree->AppendItem(mapTree->GetSelection(), map->getTitle()); | ||
363 | map->setTreeItemId(node); | ||
364 | mapTree->SetItemData(node, new MapPtrCtr(map.get())); | ||
365 | mapTree->SetFocusedItem(node); | ||
366 | mapTree->Expand(mapTree->GetSelection()); | ||
367 | SelectMap(map.get()); | ||
368 | } | ||
369 | |||
370 | void MapeditFrame::OnDidSelectMap(wxTreeEvent& event) | ||
371 | { | ||
372 | MapPtrCtr* data = (MapPtrCtr*) mapTree->GetItemData(event.GetItem()); | ||
373 | SelectMap(data->map); | ||
374 | } | ||
375 | |||
376 | void MapeditFrame::OnWillSelectMap(wxTreeEvent& event) | ||
377 | { | ||
378 | if (addingEntity) | ||
379 | { | ||
380 | event.Veto(); | ||
381 | return; | ||
382 | } | ||
383 | |||
384 | event.Skip(); | ||
385 | } | ||
386 | |||
387 | void MapeditFrame::OnWillDragMap(wxTreeEvent& event) | ||
388 | { | ||
389 | if (!addingEntity) | ||
390 | { | ||
391 | event.Allow(); | ||
392 | dragMap = event.GetItem(); | ||
393 | } | ||
394 | } | ||
395 | |||
396 | void MapeditFrame::OnDidDragMap(wxTreeEvent& event) | ||
397 | { | ||
398 | if (!dragMap.IsOk()) | ||
399 | { | ||
400 | return; | ||
401 | } | ||
402 | |||
403 | wxTreeItemId newParent = event.GetItem(); | ||
404 | if (!newParent.IsOk()) | ||
405 | { | ||
406 | newParent = mapTree->GetRootItem(); | ||
407 | } | ||
408 | |||
409 | wxTreeItemId newChild = MoveTreeNode(dragMap, newParent); | ||
410 | dragMap.Unset(); | ||
411 | mapTree->SelectItem(newChild); | ||
412 | } | ||
413 | |||
414 | void MapeditFrame::NewWorld() | ||
415 | { | ||
416 | LaunchWindow(std::unique_ptr<World>(new World())); | ||
417 | } | ||
418 | |||
419 | void MapeditFrame::OpenWorld(std::string filename) | ||
420 | { | ||
421 | try | ||
422 | { | ||
423 | auto world = std::unique_ptr<World>(new World(filename)); | ||
424 | |||
425 | LaunchWindow(std::move(world)); | ||
426 | } catch (std::exception& ex) | ||
427 | { | ||
428 | wxMessageBox(ex.what(), "Error loading world", wxOK | wxCENTRE | wxICON_ERROR); | ||
429 | } | ||
430 | } | ||
431 | |||
432 | void MapeditFrame::LaunchWindow(std::unique_ptr<World> world) | ||
433 | { | ||
434 | MapeditFrame* frame = new MapeditFrame(std::move(world)); | ||
435 | frame->closer = openWindows.insert(end(openWindows), frame); | ||
436 | frame->Show(true); | ||
437 | } | ||
438 | |||
320 | void MapeditFrame::StartAddingEntity() | 439 | void MapeditFrame::StartAddingEntity() |
321 | { | 440 | { |
322 | addingEntity = true; | 441 | addingEntity = true; |
323 | addEntityButton->Disable(); | 442 | addEntityButton->Disable(); |
324 | cancelEntityButton->Enable(); | 443 | cancelEntityButton->Enable(); |
444 | |||
445 | toolbar->EnableTool(TOOL_FILE_NEW, false); | ||
446 | toolbar->EnableTool(TOOL_FILE_OPEN, false); | ||
447 | toolbar->EnableTool(TOOL_FILE_SAVE, false); | ||
448 | toolbar->EnableTool(TOOL_MAP_ADD_ROOT, false); | ||
449 | toolbar->EnableTool(TOOL_MAP_ADD_CHILD, false); | ||
450 | |||
451 | menuFile->Enable(MENU_FILE_NEW, false); | ||
452 | menuFile->Enable(MENU_FILE_OPEN, false); | ||
453 | menuFile->Enable(MENU_FILE_SAVE, false); | ||
454 | menuFile->Enable(MENU_MAP_ADD_ROOT, false); | ||
455 | menuFile->Enable(MENU_MAP_ADD_CHILD, false); | ||
325 | } | 456 | } |
326 | 457 | ||
327 | void MapeditFrame::FinishAddingEntity() | 458 | void MapeditFrame::FinishAddingEntity() |
@@ -329,9 +460,87 @@ void MapeditFrame::FinishAddingEntity() | |||
329 | addingEntity = false; | 460 | addingEntity = false; |
330 | addEntityButton->Enable(); | 461 | addEntityButton->Enable(); |
331 | cancelEntityButton->Disable(); | 462 | cancelEntityButton->Disable(); |
463 | toolbar->Enable(); | ||
464 | |||
465 | toolbar->EnableTool(TOOL_FILE_NEW, true); | ||
466 | toolbar->EnableTool(TOOL_FILE_OPEN, true); | ||
467 | toolbar->EnableTool(TOOL_FILE_SAVE, world->getDirty()); | ||
468 | toolbar->EnableTool(TOOL_MAP_ADD_ROOT, true); | ||
469 | toolbar->EnableTool(TOOL_MAP_ADD_CHILD, true); | ||
470 | |||
471 | menuFile->Enable(MENU_FILE_NEW, true); | ||
472 | menuFile->Enable(MENU_FILE_OPEN, true); | ||
473 | menuFile->Enable(MENU_FILE_SAVE, world->getDirty()); | ||
474 | menuFile->Enable(MENU_MAP_ADD_ROOT, true); | ||
475 | menuFile->Enable(MENU_MAP_ADD_CHILD, true); | ||
332 | } | 476 | } |
333 | 477 | ||
334 | void MapeditFrame::MapDirtyDidChange(bool dirty) | 478 | void MapeditFrame::MapDirtyDidChange(bool dirty) |
335 | { | 479 | { |
336 | toolbar->EnableTool(TOOL_FILE_SAVE, dirty); | 480 | toolbar->EnableTool(TOOL_FILE_SAVE, dirty); |
481 | menuFile->Enable(MENU_FILE_SAVE, dirty); | ||
482 | |||
483 | if (dirty) | ||
484 | { | ||
485 | mapTree->SetItemBold(currentMap->getTreeItemId(), true); | ||
486 | } else { | ||
487 | for (auto map : world->getMaps()) | ||
488 | { | ||
489 | mapTree->SetItemBold(map.second->getTreeItemId(), false); | ||
490 | } | ||
491 | } | ||
492 | } | ||
493 | |||
494 | void MapeditFrame::populateMapTree(wxTreeItemId node, std::list<std::shared_ptr<Map>> maps) | ||
495 | { | ||
496 | for (auto map : maps) | ||
497 | { | ||
498 | wxTreeItemId childNode = mapTree->AppendItem(node, map->getTitle()); | ||
499 | mapTree->SetItemData(childNode, new MapPtrCtr(map.get())); | ||
500 | map->setTreeItemId(childNode); | ||
501 | |||
502 | populateMapTree(childNode, map->getChildren()); | ||
503 | |||
504 | if (map->getExpanded()) | ||
505 | { | ||
506 | mapTree->Expand(childNode); | ||
507 | } | ||
508 | } | ||
509 | } | ||
510 | |||
511 | void MapeditFrame::SelectMap(Map* map) | ||
512 | { | ||
513 | currentMap = map; | ||
514 | mapEditor->SetMap(map); | ||
515 | titleBox->ChangeValue(map->getTitle()); | ||
516 | } | ||
517 | |||
518 | wxTreeItemId MapeditFrame::MoveTreeNode(wxTreeItemId toCopy, wxTreeItemId newParent) | ||
519 | { | ||
520 | MapPtrCtr* ctl1 = (MapPtrCtr*) mapTree->GetItemData(toCopy); | ||
521 | MapPtrCtr* ctl2 = new MapPtrCtr(ctl1->map); | ||
522 | |||
523 | wxTreeItemId copied = mapTree->AppendItem(newParent, mapTree->GetItemText(toCopy), -1, -1, ctl2); | ||
524 | if (mapTree->IsBold(toCopy)) | ||
525 | { | ||
526 | mapTree->SetItemBold(toCopy, true); | ||
527 | } | ||
528 | |||
529 | if (mapTree->ItemHasChildren(toCopy)) | ||
530 | { | ||
531 | wxTreeItemIdValue cookie; | ||
532 | for (wxTreeItemId it = mapTree->GetFirstChild(toCopy, cookie); it.IsOk(); it = mapTree->GetNextChild(toCopy, cookie)) | ||
533 | { | ||
534 | MoveTreeNode(it, copied); | ||
535 | } | ||
536 | } | ||
537 | |||
538 | if (mapTree->IsExpanded(toCopy)) | ||
539 | { | ||
540 | mapTree->Expand(copied); | ||
541 | } | ||
542 | |||
543 | mapTree->Delete(toCopy); | ||
544 | |||
545 | return copied; | ||
337 | } | 546 | } |
diff --git a/tools/mapedit/src/frame.h b/tools/mapedit/src/frame.h index 4aa44c5..f1e4efc 100644 --- a/tools/mapedit/src/frame.h +++ b/tools/mapedit/src/frame.h | |||
@@ -12,24 +12,36 @@ | |||
12 | #include "tile_widget.h" | 12 | #include "tile_widget.h" |
13 | #include <list> | 13 | #include <list> |
14 | #include <wx/notebook.h> | 14 | #include <wx/notebook.h> |
15 | #include <memory> | ||
16 | #include <wx/treectrl.h> | ||
17 | |||
18 | class MapPtrCtr : public wxTreeItemData { | ||
19 | public: | ||
20 | Map* map; | ||
21 | |||
22 | MapPtrCtr(Map* map) : map(map) {} | ||
23 | }; | ||
15 | 24 | ||
16 | class MapeditFrame : public wxFrame { | 25 | class MapeditFrame : public wxFrame { |
17 | public: | 26 | public: |
18 | MapeditFrame() {} | 27 | MapeditFrame() {} |
19 | MapeditFrame(Map map, std::string filename); | 28 | MapeditFrame(std::unique_ptr<World> world); |
20 | 29 | ||
21 | MapeditWidget* GetMapEditor(); | 30 | MapeditWidget* GetMapEditor(); |
22 | void StartAddingEntity(); | 31 | void StartAddingEntity(); |
23 | void FinishAddingEntity(); | 32 | void FinishAddingEntity(); |
24 | void MapDirtyDidChange(bool dirty); | 33 | void MapDirtyDidChange(bool dirty); |
25 | 34 | ||
26 | static void NewMap(); | 35 | static void NewWorld(); |
27 | static void OpenMap(const char* filename); | 36 | static void OpenWorld(std::string filename); |
28 | 37 | ||
29 | std::list<wxWindow*>::iterator closer; | 38 | std::list<wxWindow*>::iterator closer; |
30 | 39 | ||
31 | private: | 40 | private: |
32 | static void LaunchWindow(Map map, const char* filename); | 41 | static void LaunchWindow(std::unique_ptr<World> world); |
42 | void populateMapTree(wxTreeItemId node, std::list<std::shared_ptr<Map>> maps); | ||
43 | void SelectMap(Map* map); | ||
44 | wxTreeItemId MoveTreeNode(wxTreeItemId toCopy, wxTreeItemId newParent); | ||
33 | 45 | ||
34 | void ZoomIn(wxCommandEvent& event); | 46 | void ZoomIn(wxCommandEvent& event); |
35 | void ZoomOut(wxCommandEvent& event); | 47 | void ZoomOut(wxCommandEvent& event); |
@@ -44,8 +56,15 @@ class MapeditFrame : public wxFrame { | |||
44 | void OnTabChanging(wxBookCtrlEvent& event); | 56 | void OnTabChanging(wxBookCtrlEvent& event); |
45 | void OnAddEntity(wxCommandEvent& event); | 57 | void OnAddEntity(wxCommandEvent& event); |
46 | void OnCancelAddEntity(wxCommandEvent& event); | 58 | void OnCancelAddEntity(wxCommandEvent& event); |
59 | void OnAddRoot(wxCommandEvent& event); | ||
60 | void OnAddChild(wxCommandEvent& event); | ||
61 | void OnDidSelectMap(wxTreeEvent& event); | ||
62 | void OnWillSelectMap(wxTreeEvent& event); | ||
63 | void OnWillDragMap(wxTreeEvent& event); | ||
64 | void OnDidDragMap(wxTreeEvent& event); | ||
47 | 65 | ||
48 | Map map; | 66 | std::unique_ptr<World> world; |
67 | Map* currentMap; | ||
49 | MapeditWidget* mapEditor; | 68 | MapeditWidget* mapEditor; |
50 | TileWidget* tileEditor; | 69 | TileWidget* tileEditor; |
51 | wxTextCtrl* titleBox; | 70 | wxTextCtrl* titleBox; |
@@ -55,6 +74,9 @@ class MapeditFrame : public wxFrame { | |||
55 | wxButton* addEntityButton; | 74 | wxButton* addEntityButton; |
56 | wxButton* cancelEntityButton; | 75 | wxButton* cancelEntityButton; |
57 | wxToolBar* toolbar; | 76 | wxToolBar* toolbar; |
77 | wxMenu* menuFile; | ||
78 | wxTreeCtrl* mapTree; | ||
79 | wxTreeItemId dragMap; | ||
58 | 80 | ||
59 | bool addingEntity = false; | 81 | bool addingEntity = false; |
60 | 82 | ||
diff --git a/tools/mapedit/src/main.cpp b/tools/mapedit/src/main.cpp index 91dd98f..b09ee9a 100644 --- a/tools/mapedit/src/main.cpp +++ b/tools/mapedit/src/main.cpp | |||
@@ -18,7 +18,7 @@ bool MapeditApp::OnInit() | |||
18 | { | 18 | { |
19 | wxInitAllImageHandlers(); | 19 | wxInitAllImageHandlers(); |
20 | 20 | ||
21 | MapeditFrame::NewMap(); | 21 | MapeditFrame::NewWorld(); |
22 | 22 | ||
23 | return true; | 23 | return true; |
24 | } | 24 | } |
diff --git a/tools/mapedit/src/map.cpp b/tools/mapedit/src/map.cpp index f9c07fc..32541e6 100644 --- a/tools/mapedit/src/map.cpp +++ b/tools/mapedit/src/map.cpp | |||
@@ -1,106 +1,27 @@ | |||
1 | #include "map.h" | 1 | #include "map.h" |
2 | #include <libxml/parser.h> | ||
3 | #include <libxml/xmlwriter.h> | ||
4 | #include <sstream> | ||
5 | #include "frame.h" | 2 | #include "frame.h" |
6 | 3 | ||
7 | Map::Map() | 4 | Map::Map(int id, World* world) : id(id), world(world) |
8 | { | 5 | { |
9 | mapdata = (int*) calloc(MAP_WIDTH * MAP_HEIGHT, sizeof(int)); | 6 | mapdata = (int*) calloc(MAP_WIDTH * MAP_HEIGHT, sizeof(int)); |
10 | } | 7 | } |
11 | 8 | ||
12 | Map::Map(std::string filename) | ||
13 | { | ||
14 | xmlDocPtr doc = xmlParseFile(filename.c_str()); | ||
15 | if (doc == nullptr) | ||
16 | { | ||
17 | throw MapLoadException(filename); | ||
18 | } | ||
19 | |||
20 | xmlNodePtr top = xmlDocGetRootElement(doc); | ||
21 | if (top == nullptr) | ||
22 | { | ||
23 | throw MapLoadException(filename); | ||
24 | } | ||
25 | |||
26 | if (xmlStrcmp(top->name, (const xmlChar*) "map-def")) | ||
27 | { | ||
28 | throw MapLoadException(filename); | ||
29 | } | ||
30 | |||
31 | for (xmlNodePtr node = top->xmlChildrenNode; node != NULL; node = node->next) | ||
32 | { | ||
33 | if (!xmlStrcmp(node->name, (const xmlChar*) "name")) | ||
34 | { | ||
35 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
36 | title = (char*) key; | ||
37 | xmlFree(key); | ||
38 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "environment")) | ||
39 | { | ||
40 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
41 | mapdata = (int*) malloc(MAP_WIDTH*MAP_HEIGHT*sizeof(int)); | ||
42 | mapdata[0] = atoi(strtok((char*) key, ",\n")); | ||
43 | for (int i=1; i<(MAP_WIDTH*MAP_HEIGHT); i++) | ||
44 | { | ||
45 | mapdata[i] = atoi(strtok(NULL, ",\n")); | ||
46 | } | ||
47 | xmlFree(key); | ||
48 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "leftmap")) | ||
49 | { | ||
50 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
51 | leftmap = (char*) key; | ||
52 | xmlFree(key); | ||
53 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "rightmap")) | ||
54 | { | ||
55 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
56 | rightmap = (char*) key; | ||
57 | xmlFree(key); | ||
58 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "entities")) | ||
59 | { | ||
60 | for (xmlNodePtr entityNode = node->xmlChildrenNode; entityNode != NULL; entityNode = entityNode->next) | ||
61 | { | ||
62 | if (!xmlStrcmp(entityNode->name, (const xmlChar*) "entity")) | ||
63 | { | ||
64 | auto data = std::make_shared<MapObjectEntry>(); | ||
65 | |||
66 | for (xmlNodePtr entityDataNode = entityNode->xmlChildrenNode; entityDataNode != NULL; entityDataNode = entityDataNode->next) | ||
67 | { | ||
68 | if (!xmlStrcmp(entityDataNode->name, (const xmlChar*) "entity-type")) | ||
69 | { | ||
70 | xmlChar* key = xmlNodeListGetString(doc, entityDataNode->xmlChildrenNode, 1); | ||
71 | data->object = MapObject::getAllObjects().at((char*) key).get(); | ||
72 | xmlFree(key); | ||
73 | } else if (!xmlStrcmp(entityDataNode->name, (const xmlChar*) "entity-position")) | ||
74 | { | ||
75 | xmlChar* key = xmlNodeListGetString(doc, entityDataNode->xmlChildrenNode, 1); | ||
76 | sscanf((char*) key, "%lf,%lf", &data->position.first, &data->position.second); | ||
77 | xmlFree(key); | ||
78 | } | ||
79 | } | ||
80 | |||
81 | objects.push_back(data); | ||
82 | } | ||
83 | } | ||
84 | } | ||
85 | } | ||
86 | |||
87 | xmlFreeDoc(doc); | ||
88 | } | ||
89 | |||
90 | Map::Map(const Map& map) | 9 | Map::Map(const Map& map) |
91 | { | 10 | { |
92 | mapdata = (int*) malloc(MAP_WIDTH*MAP_HEIGHT*sizeof(int)); | 11 | mapdata = (int*) malloc(MAP_WIDTH*MAP_HEIGHT*sizeof(int)); |
93 | memcpy(mapdata, map.mapdata, MAP_WIDTH*MAP_HEIGHT*sizeof(int)); | 12 | memcpy(mapdata, map.mapdata, MAP_WIDTH*MAP_HEIGHT*sizeof(int)); |
94 | 13 | ||
14 | id = map.id; | ||
95 | title = map.title; | 15 | title = map.title; |
96 | leftmap = map.leftmap; | 16 | leftmap = map.leftmap; |
97 | rightmap = map.rightmap; | 17 | rightmap = map.rightmap; |
98 | dirty = map.dirty; | ||
99 | objects = map.objects; | 18 | objects = map.objects; |
100 | frame = map.frame; | 19 | world = map.world; |
20 | treeItemId = map.treeItemId; | ||
21 | children = map.children; | ||
101 | } | 22 | } |
102 | 23 | ||
103 | Map::Map(Map&& map) : Map() | 24 | Map::Map(Map&& map) : Map(-1, map.world) |
104 | { | 25 | { |
105 | swap(*this, map); | 26 | swap(*this, map); |
106 | } | 27 | } |
@@ -123,141 +44,157 @@ void swap(Map& first, Map& second) | |||
123 | std::swap(first.title, second.title); | 44 | std::swap(first.title, second.title); |
124 | std::swap(first.leftmap, second.leftmap); | 45 | std::swap(first.leftmap, second.leftmap); |
125 | std::swap(first.rightmap, second.rightmap); | 46 | std::swap(first.rightmap, second.rightmap); |
126 | std::swap(first.dirty, second.dirty); | ||
127 | std::swap(first.objects, second.objects); | 47 | std::swap(first.objects, second.objects); |
128 | std::swap(first.frame, second.frame); | 48 | std::swap(first.id, second.id); |
49 | std::swap(first.world, second.world); | ||
50 | std::swap(first.treeItemId, second.treeItemId); | ||
51 | std::swap(first.children, second.children); | ||
129 | } | 52 | } |
130 | 53 | ||
131 | #define MY_ENCODING "ISO-8859-1" | 54 | int Map::getID() const |
55 | { | ||
56 | return id; | ||
57 | } | ||
132 | 58 | ||
133 | void Map::save(std::string name) | 59 | std::string Map::getTitle() const |
134 | { | 60 | { |
135 | if (!dirty) return; | 61 | return title; |
136 | 62 | } | |
137 | int rc; | ||
138 | |||
139 | xmlTextWriterPtr writer = xmlNewTextWriterFilename(name.c_str(), 0); | ||
140 | if (writer == NULL) throw MapWriteException(name); | ||
141 | 63 | ||
142 | rc = xmlTextWriterStartDocument(writer, NULL, MY_ENCODING, NULL); | 64 | int Map::getTileAt(int x, int y) const |
143 | if (rc < 0) throw MapWriteException(name); | 65 | { |
144 | 66 | return mapdata[x+y*MAP_WIDTH]; | |
145 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "map-def"); | 67 | } |
146 | if (rc < 0) throw MapWriteException(name); | 68 | |
147 | 69 | const std::list<std::shared_ptr<MapObjectEntry>>& Map::getObjects() const | |
148 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "name", (xmlChar*) title.c_str()); | 70 | { |
149 | if (rc < 0) throw MapWriteException(name); | 71 | return objects; |
150 | 72 | } | |
151 | std::ostringstream mapdata_out; | 73 | |
152 | for (int y=0; y<MAP_HEIGHT; y++) | 74 | std::shared_ptr<Map> Map::getLeftmap() const |
75 | { | ||
76 | if (leftmap == -1) | ||
153 | { | 77 | { |
154 | for (int x=0; x<MAP_WIDTH; x++) | 78 | return std::shared_ptr<Map>(); |
155 | { | 79 | } else { |
156 | mapdata_out << mapdata[x+y*MAP_WIDTH] << ","; | 80 | return world->getMap(leftmap); |
157 | } | ||
158 | |||
159 | mapdata_out << std::endl; | ||
160 | } | 81 | } |
161 | 82 | } | |
162 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "environment", (xmlChar*) mapdata_out.str().c_str()); | ||
163 | if (rc < 0) throw MapWriteException(name); | ||
164 | |||
165 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "leftmap", (xmlChar*) leftmap.c_str()); | ||
166 | if (rc < 0) throw MapWriteException(name); | ||
167 | 83 | ||
168 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "rightmap", (xmlChar*) rightmap.c_str()); | 84 | std::shared_ptr<Map> Map::getRightmap() const |
169 | if (rc < 0) throw MapWriteException(name); | 85 | { |
170 | 86 | if (rightmap == -1) | |
171 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "entities"); | ||
172 | if (rc < 0) throw MapWriteException(name); | ||
173 | |||
174 | for (auto object : objects) | ||
175 | { | 87 | { |
176 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "entity"); | 88 | return std::shared_ptr<Map>(); |
177 | if (rc < 0) throw MapWriteException(name); | 89 | } else { |
178 | 90 | return world->getMap(rightmap); | |
179 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "entity-type", (xmlChar*) object->object->getType().c_str()); | ||
180 | if (rc < 0) throw MapWriteException(name); | ||
181 | |||
182 | std::ostringstream entpos_out; | ||
183 | entpos_out << object->position.first << "," << object->position.second; | ||
184 | |||
185 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "entity-position", (xmlChar*) entpos_out.str().c_str()); | ||
186 | if (rc < 0) throw MapWriteException(name); | ||
187 | |||
188 | rc = xmlTextWriterEndElement(writer); | ||
189 | if (rc < 0) throw MapWriteException(name); | ||
190 | } | 91 | } |
191 | |||
192 | rc = xmlTextWriterEndElement(writer); | ||
193 | if (rc < 0) throw MapWriteException(name); | ||
194 | |||
195 | rc = xmlTextWriterEndElement(writer); | ||
196 | if (rc < 0) throw MapWriteException(name); | ||
197 | |||
198 | rc = xmlTextWriterEndDocument(writer); | ||
199 | if (rc < 0) throw MapWriteException(name); | ||
200 | |||
201 | xmlFreeTextWriter(writer); | ||
202 | |||
203 | setDirty(false); | ||
204 | } | 92 | } |
205 | 93 | ||
206 | bool Map::hasUnsavedChanges() const | 94 | wxTreeItemId Map::getTreeItemId() const |
207 | { | 95 | { |
208 | return dirty; | 96 | return treeItemId; |
209 | } | 97 | } |
210 | 98 | ||
211 | void Map::setTileAt(int x, int y, int tile) | 99 | std::list<std::shared_ptr<Map>> Map::getChildren() const |
212 | { | 100 | { |
213 | setDirty(true); | 101 | std::list<std::shared_ptr<Map>> ret; |
214 | mapdata[x+y*MAP_WIDTH] = tile; | 102 | |
103 | for (auto id : children) | ||
104 | { | ||
105 | ret.push_back(world->getMap(id)); | ||
106 | } | ||
107 | |||
108 | return ret; | ||
215 | } | 109 | } |
216 | 110 | ||
217 | int Map::getTileAt(int x, int y) const | 111 | bool Map::getExpanded() const |
218 | { | 112 | { |
219 | return mapdata[x+y*MAP_WIDTH]; | 113 | return expanded; |
220 | } | 114 | } |
221 | 115 | ||
222 | std::string Map::getTitle() const | 116 | void Map::setTitle(std::string title, bool dirty) |
223 | { | 117 | { |
224 | return title; | 118 | this->title = title; |
119 | |||
120 | if (dirty) | ||
121 | { | ||
122 | world->setDirty(true); | ||
123 | } | ||
225 | } | 124 | } |
226 | 125 | ||
227 | void Map::setTitle(std::string title) | 126 | void Map::setTileAt(int x, int y, int tile, bool dirty) |
228 | { | 127 | { |
229 | setDirty(true); | 128 | mapdata[x+y*MAP_WIDTH] = tile; |
230 | this->title = title; | 129 | |
130 | if (dirty) | ||
131 | { | ||
132 | world->setDirty(true); | ||
133 | } | ||
231 | } | 134 | } |
232 | 135 | ||
233 | const std::list<std::shared_ptr<MapObjectEntry>>& Map::getObjects() const | 136 | void Map::setMapdata(int* mapdata, bool dirty) |
234 | { | 137 | { |
235 | return objects; | 138 | free(this->mapdata); |
139 | this->mapdata = mapdata; | ||
140 | |||
141 | if (dirty) | ||
142 | { | ||
143 | world->setDirty(true); | ||
144 | } | ||
236 | } | 145 | } |
237 | 146 | ||
238 | void Map::addObject(std::shared_ptr<MapObjectEntry>& obj) | 147 | void Map::addObject(std::shared_ptr<MapObjectEntry>& obj, bool dirty) |
239 | { | 148 | { |
240 | setDirty(true); | ||
241 | objects.push_back(obj); | 149 | objects.push_back(obj); |
150 | |||
151 | if (dirty) | ||
152 | { | ||
153 | world->setDirty(true); | ||
154 | } | ||
242 | } | 155 | } |
243 | 156 | ||
244 | void Map::removeObject(std::shared_ptr<MapObjectEntry>& obj) | 157 | void Map::removeObject(std::shared_ptr<MapObjectEntry>& obj, bool dirty) |
245 | { | 158 | { |
246 | setDirty(true); | ||
247 | objects.remove(obj); | 159 | objects.remove(obj); |
160 | |||
161 | if (dirty) | ||
162 | { | ||
163 | world->setDirty(true); | ||
164 | } | ||
248 | } | 165 | } |
249 | 166 | ||
250 | bool Map::getDirty() const | 167 | void Map::setLeftmap(int id, bool dirty) |
251 | { | 168 | { |
252 | return dirty; | 169 | leftmap = id; |
170 | |||
171 | if (dirty) | ||
172 | { | ||
173 | world->setDirty(true); | ||
174 | } | ||
253 | } | 175 | } |
254 | 176 | ||
255 | void Map::setDirty(bool dirty) | 177 | void Map::setRightmap(int id, bool dirty) |
256 | { | 178 | { |
257 | this->dirty = dirty; | 179 | rightmap = id; |
258 | 180 | ||
259 | if (frame != nullptr) | 181 | if (dirty) |
260 | { | 182 | { |
261 | frame->MapDirtyDidChange(dirty); | 183 | world->setDirty(true); |
262 | } | 184 | } |
263 | } | 185 | } |
186 | |||
187 | void Map::setTreeItemId(wxTreeItemId id) | ||
188 | { | ||
189 | this->treeItemId = id; | ||
190 | } | ||
191 | |||
192 | void Map::addChild(int id) | ||
193 | { | ||
194 | children.push_back(id); | ||
195 | } | ||
196 | |||
197 | void Map::setExpanded(bool exp) | ||
198 | { | ||
199 | expanded = exp; | ||
200 | } | ||
diff --git a/tools/mapedit/src/map.h b/tools/mapedit/src/map.h index a2b9cb8..5753cae 100644 --- a/tools/mapedit/src/map.h +++ b/tools/mapedit/src/map.h | |||
@@ -1,12 +1,16 @@ | |||
1 | #ifndef MAP_H | 1 | #ifndef MAP_H |
2 | #define MAP_H | 2 | #define MAP_H |
3 | 3 | ||
4 | class Map; | ||
5 | |||
4 | #include <string> | 6 | #include <string> |
5 | #include <exception> | 7 | #include <exception> |
6 | #include <utility> | 8 | #include <utility> |
7 | #include <list> | 9 | #include <list> |
8 | #include "object.h" | 10 | #include "object.h" |
9 | #include <memory> | 11 | #include <memory> |
12 | #include "world.h" | ||
13 | #include <wx/treectrl.h> | ||
10 | 14 | ||
11 | class MapeditFrame; | 15 | class MapeditFrame; |
12 | 16 | ||
@@ -57,36 +61,45 @@ struct MapObjectEntry { | |||
57 | 61 | ||
58 | class Map { | 62 | class Map { |
59 | public: | 63 | public: |
60 | Map(); | 64 | Map(int id, World* world); |
61 | Map(std::string name); | ||
62 | Map(const Map& map); | 65 | Map(const Map& map); |
63 | Map(Map&& map); | 66 | Map(Map&& map); |
64 | ~Map(); | 67 | ~Map(); |
65 | Map& operator= (Map other); | 68 | Map& operator= (Map other); |
66 | friend void swap(Map& first, Map& second); | 69 | friend void swap(Map& first, Map& second); |
67 | 70 | ||
71 | int getID() const; | ||
68 | std::string getTitle() const; | 72 | std::string getTitle() const; |
69 | void setTitle(std::string title); | ||
70 | void save(std::string name); | ||
71 | bool hasUnsavedChanges() const; | ||
72 | void setTileAt(int x, int y, int tile); | ||
73 | int getTileAt(int x, int y) const; | 73 | int getTileAt(int x, int y) const; |
74 | const std::list<std::shared_ptr<MapObjectEntry>>& getObjects() const; | 74 | const std::list<std::shared_ptr<MapObjectEntry>>& getObjects() const; |
75 | void addObject(std::shared_ptr<MapObjectEntry>& obj); | 75 | std::shared_ptr<Map> getLeftmap() const; |
76 | void removeObject(std::shared_ptr<MapObjectEntry>& obj); | 76 | std::shared_ptr<Map> getRightmap() const; |
77 | bool getDirty() const; | 77 | wxTreeItemId getTreeItemId() const; |
78 | std::list<std::shared_ptr<Map>> getChildren() const; | ||
79 | bool getExpanded() const; | ||
78 | 80 | ||
79 | MapeditFrame* frame; | 81 | void setTitle(std::string title, bool dirty = true); |
82 | void setTileAt(int x, int y, int tile, bool dirty = true); | ||
83 | void setMapdata(int* mapdata, bool dirty = true); | ||
84 | void addObject(std::shared_ptr<MapObjectEntry>& obj, bool dirty = true); | ||
85 | void removeObject(std::shared_ptr<MapObjectEntry>& obj, bool dirty = true); | ||
86 | void setLeftmap(int id, bool dirty = true); | ||
87 | void setRightmap(int id, bool dirty = true); | ||
88 | void setTreeItemId(wxTreeItemId id); | ||
89 | void addChild(int id); | ||
90 | void setExpanded(bool exp); | ||
80 | 91 | ||
81 | private: | 92 | private: |
82 | void setDirty(bool dirty); | 93 | int id; |
83 | 94 | World* world; | |
84 | std::list<std::shared_ptr<MapObjectEntry>> objects; | 95 | std::list<std::shared_ptr<MapObjectEntry>> objects; |
85 | int* mapdata; | 96 | int* mapdata; |
86 | std::string title; | 97 | std::string title {"Untitled Map"}; |
87 | std::string leftmap; | 98 | std::list<int> children; |
88 | std::string rightmap; | 99 | int leftmap = -1; |
89 | bool dirty; | 100 | int rightmap = -1; |
101 | wxTreeItemId treeItemId; | ||
102 | bool expanded = false; | ||
90 | }; | 103 | }; |
91 | 104 | ||
92 | #endif | 105 | #endif |
diff --git a/tools/mapedit/src/widget.cpp b/tools/mapedit/src/widget.cpp index c1f044c..d50cf91 100644 --- a/tools/mapedit/src/widget.cpp +++ b/tools/mapedit/src/widget.cpp | |||
@@ -297,3 +297,10 @@ void MapeditWidget::CancelAddingEntity() | |||
297 | { | 297 | { |
298 | addingEntity = nullptr; | 298 | addingEntity = nullptr; |
299 | } | 299 | } |
300 | |||
301 | void MapeditWidget::SetMap(Map* map) | ||
302 | { | ||
303 | this->map = map; | ||
304 | |||
305 | Refresh(); | ||
306 | } | ||
diff --git a/tools/mapedit/src/widget.h b/tools/mapedit/src/widget.h index a660f82..34627bc 100644 --- a/tools/mapedit/src/widget.h +++ b/tools/mapedit/src/widget.h | |||
@@ -29,6 +29,7 @@ class MapeditWidget : public wxScrolledWindow { | |||
29 | void SetEditMode(EditMode editMode); | 29 | void SetEditMode(EditMode editMode); |
30 | void StartAddingEntity(MapObject* object); | 30 | void StartAddingEntity(MapObject* object); |
31 | void CancelAddingEntity(); | 31 | void CancelAddingEntity(); |
32 | void SetMap(Map* map); | ||
32 | 33 | ||
33 | MapeditFrame* frame; | 34 | MapeditFrame* frame; |
34 | 35 | ||
@@ -46,7 +47,7 @@ class MapeditWidget : public wxScrolledWindow { | |||
46 | void SetTile(wxPoint pos); | 47 | void SetTile(wxPoint pos); |
47 | void SetZoomSize(int zoom); | 48 | void SetZoomSize(int zoom); |
48 | 49 | ||
49 | Map* const map = nullptr; | 50 | Map* map = nullptr; |
50 | wxBitmap tiles; | 51 | wxBitmap tiles; |
51 | TileWidget* tileWidget; | 52 | TileWidget* tileWidget; |
52 | bool mouseIsDown = false; | 53 | bool mouseIsDown = false; |
diff --git a/tools/mapedit/src/world.cpp b/tools/mapedit/src/world.cpp new file mode 100644 index 0000000..4c42593 --- /dev/null +++ b/tools/mapedit/src/world.cpp | |||
@@ -0,0 +1,369 @@ | |||
1 | #include "world.h" | ||
2 | #include <libxml/parser.h> | ||
3 | #include <libxml/xmlwriter.h> | ||
4 | #include "frame.h" | ||
5 | #include <sstream> | ||
6 | |||
7 | World::World() | ||
8 | { | ||
9 | newMap(); | ||
10 | |||
11 | rootChildren.push_back(0); | ||
12 | } | ||
13 | |||
14 | World::World(std::string filename) | ||
15 | { | ||
16 | this->filename = filename; | ||
17 | |||
18 | xmlDocPtr doc = xmlParseFile(filename.c_str()); | ||
19 | if (doc == nullptr) | ||
20 | { | ||
21 | throw MapLoadException(filename); | ||
22 | } | ||
23 | |||
24 | xmlNodePtr top = xmlDocGetRootElement(doc); | ||
25 | if (top == nullptr) | ||
26 | { | ||
27 | throw MapLoadException(filename); | ||
28 | } | ||
29 | |||
30 | if (xmlStrcmp(top->name, (const xmlChar*) "world")) | ||
31 | { | ||
32 | throw MapLoadException(filename); | ||
33 | } | ||
34 | |||
35 | for (xmlNodePtr node = top->xmlChildrenNode; node != NULL; node = node->next) | ||
36 | { | ||
37 | if (!xmlStrcmp(node->name, (const xmlChar*) "nextmapid")) | ||
38 | { | ||
39 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
40 | if (key != 0) | ||
41 | { | ||
42 | nextMapID = atoi((char*) key); | ||
43 | } | ||
44 | xmlFree(key); | ||
45 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "lastmap")) | ||
46 | { | ||
47 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
48 | if (key != 0) | ||
49 | { | ||
50 | lastmap = atoi((char*) key); | ||
51 | } | ||
52 | xmlFree(key); | ||
53 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "root")) | ||
54 | { | ||
55 | xmlChar* key = xmlNodeListGetString(doc, node->xmlChildrenNode, 1); | ||
56 | if (key != 0) | ||
57 | { | ||
58 | rootChildren.push_back(atoi((char*) key)); | ||
59 | } | ||
60 | xmlFree(key); | ||
61 | } else if (!xmlStrcmp(node->name, (const xmlChar*) "map")) | ||
62 | { | ||
63 | xmlChar* idKey = xmlGetProp(node, (xmlChar*) "id"); | ||
64 | if (idKey == 0) throw MapLoadException(filename); | ||
65 | int id = atoi((char*) idKey); | ||
66 | xmlFree(idKey); | ||
67 | |||
68 | auto map = std::make_shared<Map>(id, this); | ||
69 | |||
70 | for (xmlNodePtr mapNode = node->xmlChildrenNode; mapNode != NULL; mapNode = mapNode->next) | ||
71 | { | ||
72 | if (!xmlStrcmp(mapNode->name, (const xmlChar*) "name")) | ||
73 | { | ||
74 | xmlChar* key = xmlNodeListGetString(doc, mapNode->xmlChildrenNode, 1); | ||
75 | if (key != 0) | ||
76 | { | ||
77 | map->setTitle((char*) key, false); | ||
78 | } | ||
79 | |||
80 | xmlFree(key); | ||
81 | } else if (!xmlStrcmp(mapNode->name, (const xmlChar*) "environment")) | ||
82 | { | ||
83 | xmlChar* key = xmlNodeListGetString(doc, mapNode->xmlChildrenNode, 1); | ||
84 | int* mapdata = (int*) malloc(MAP_WIDTH*MAP_HEIGHT*sizeof(int)); | ||
85 | mapdata[0] = atoi(strtok((char*) key, ",\n")); | ||
86 | for (int i=1; i<(MAP_WIDTH*MAP_HEIGHT); i++) | ||
87 | { | ||
88 | mapdata[i] = atoi(strtok(NULL, ",\n")); | ||
89 | } | ||
90 | map->setMapdata(mapdata, false); | ||
91 | xmlFree(key); | ||
92 | } else if (!xmlStrcmp(mapNode->name, (const xmlChar*) "leftmap")) | ||
93 | { | ||
94 | xmlChar* key = xmlNodeListGetString(doc, mapNode->xmlChildrenNode, 1); | ||
95 | if (key != 0) | ||
96 | { | ||
97 | map->setLeftmap(atoi((char*) key), false); | ||
98 | } | ||
99 | xmlFree(key); | ||
100 | } else if (!xmlStrcmp(mapNode->name, (const xmlChar*) "rightmap")) | ||
101 | { | ||
102 | xmlChar* key = xmlNodeListGetString(doc, mapNode->xmlChildrenNode, 1); | ||
103 | if (key != 0) | ||
104 | { | ||
105 | map->setRightmap(atoi((char*) key)); | ||
106 | } | ||
107 | xmlFree(key); | ||
108 | } else if (!xmlStrcmp(mapNode->name, (const xmlChar*) "entities")) | ||
109 | { | ||
110 | for (xmlNodePtr entityNode = mapNode->xmlChildrenNode; entityNode != NULL; entityNode = entityNode->next) | ||
111 | { | ||
112 | if (!xmlStrcmp(entityNode->name, (const xmlChar*) "entity")) | ||
113 | { | ||
114 | auto data = std::make_shared<MapObjectEntry>(); | ||
115 | |||
116 | for (xmlNodePtr entityDataNode = entityNode->xmlChildrenNode; entityDataNode != NULL; entityDataNode = entityDataNode->next) | ||
117 | { | ||
118 | if (!xmlStrcmp(entityDataNode->name, (const xmlChar*) "entity-type")) | ||
119 | { | ||
120 | xmlChar* key = xmlNodeListGetString(doc, entityDataNode->xmlChildrenNode, 1); | ||
121 | data->object = MapObject::getAllObjects().at((char*) key).get(); | ||
122 | xmlFree(key); | ||
123 | } else if (!xmlStrcmp(entityDataNode->name, (const xmlChar*) "entity-position")) | ||
124 | { | ||
125 | xmlChar* key = xmlNodeListGetString(doc, entityDataNode->xmlChildrenNode, 1); | ||
126 | sscanf((char*) key, "%lf,%lf", &data->position.first, &data->position.second); | ||
127 | xmlFree(key); | ||
128 | } | ||
129 | } | ||
130 | |||
131 | map->addObject(data, false); | ||
132 | } | ||
133 | } | ||
134 | } else if (!xmlStrcmp(mapNode->name, (const xmlChar*) "child")) | ||
135 | { | ||
136 | xmlChar* key = xmlNodeListGetString(doc, mapNode->xmlChildrenNode, 1); | ||
137 | if (key != 0) | ||
138 | { | ||
139 | map->addChild(atoi((char*) key)); | ||
140 | } | ||
141 | xmlFree(key); | ||
142 | } else if (!xmlStrcmp(mapNode->name, (const xmlChar*) "expanded")) | ||
143 | { | ||
144 | xmlChar* key = xmlNodeListGetString(doc, mapNode->xmlChildrenNode, 1); | ||
145 | if ((key != 0) && ((char) key[0] == '1')) | ||
146 | { | ||
147 | map->setExpanded(true); | ||
148 | } | ||
149 | } | ||
150 | } | ||
151 | |||
152 | maps[map->getID()] = map; | ||
153 | } | ||
154 | } | ||
155 | |||
156 | xmlFreeDoc(doc); | ||
157 | } | ||
158 | |||
159 | std::shared_ptr<Map> World::newMap() | ||
160 | { | ||
161 | auto nm = std::make_shared<Map>(nextMapID++, this); | ||
162 | maps[nm->getID()] = nm; | ||
163 | return nm; | ||
164 | } | ||
165 | |||
166 | std::shared_ptr<Map> World::getMap(int id) const | ||
167 | { | ||
168 | return maps.at(id); | ||
169 | } | ||
170 | |||
171 | void World::setDirty(bool dirty) | ||
172 | { | ||
173 | this->dirty = dirty; | ||
174 | parent->MapDirtyDidChange(dirty); | ||
175 | } | ||
176 | |||
177 | bool World::getDirty() const | ||
178 | { | ||
179 | return dirty; | ||
180 | } | ||
181 | |||
182 | std::string World::getFilename() const | ||
183 | { | ||
184 | return filename; | ||
185 | } | ||
186 | |||
187 | void World::setParent(MapeditFrame* parent) | ||
188 | { | ||
189 | this->parent = parent; | ||
190 | } | ||
191 | |||
192 | Map* World::getLastMap() const | ||
193 | { | ||
194 | return getMap(lastmap).get(); | ||
195 | } | ||
196 | |||
197 | #define MY_ENCODING "ISO-8859-1" | ||
198 | |||
199 | void World::save(std::string name, wxTreeCtrl* mapTree) | ||
200 | { | ||
201 | int rc; | ||
202 | |||
203 | xmlTextWriterPtr writer = xmlNewTextWriterFilename(name.c_str(), 0); | ||
204 | if (writer == NULL) throw MapWriteException(name); | ||
205 | |||
206 | rc = xmlTextWriterStartDocument(writer, NULL, MY_ENCODING, NULL); | ||
207 | if (rc < 0) throw MapWriteException(name); | ||
208 | |||
209 | // <world> | ||
210 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "world"); | ||
211 | if (rc < 0) throw MapWriteException(name); | ||
212 | |||
213 | // <nextmapid/> | ||
214 | std::ostringstream nextMap_out; | ||
215 | nextMap_out << nextMapID; | ||
216 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "nextmapid", (xmlChar*) nextMap_out.str().c_str()); | ||
217 | if (rc < 0) throw MapWriteException(name); | ||
218 | |||
219 | // <lastmap/> | ||
220 | std::ostringstream lastMap_out; | ||
221 | lastMap_out << lastmap; | ||
222 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "lastmap", (xmlChar*) lastMap_out.str().c_str()); | ||
223 | if (rc < 0) throw MapWriteException(name); | ||
224 | |||
225 | // ASSUMPTION: There will always be at least one child of the invisible root element. i.e. you cannot delete to zero maps. | ||
226 | wxTreeItemId root = mapTree->GetRootItem(); | ||
227 | wxTreeItemIdValue cookie1; | ||
228 | for (wxTreeItemId it = mapTree->GetFirstChild(root, cookie1); it.IsOk(); it = mapTree->GetNextChild(root, cookie1)) | ||
229 | { | ||
230 | // <root> | ||
231 | MapPtrCtr* ctl = (MapPtrCtr*) mapTree->GetItemData(it); | ||
232 | std::ostringstream rootid_out; | ||
233 | rootid_out << ctl->map->getID(); | ||
234 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "root", (xmlChar*) rootid_out.str().c_str()); | ||
235 | if (rc < 0) throw MapWriteException(name); | ||
236 | } | ||
237 | |||
238 | for (auto mapPair : maps) | ||
239 | { | ||
240 | Map& map = *mapPair.second; | ||
241 | |||
242 | // <map> | ||
243 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "map"); | ||
244 | if (rc < 0) throw MapWriteException(name); | ||
245 | |||
246 | // id= | ||
247 | std::ostringstream id_out; | ||
248 | id_out << map.getID(); | ||
249 | rc = xmlTextWriterWriteAttribute(writer, (xmlChar*) "id", (xmlChar*) id_out.str().c_str()); | ||
250 | if (rc < 0) throw MapWriteException(name); | ||
251 | |||
252 | // <name/> | ||
253 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "name", (xmlChar*) map.getTitle().c_str()); | ||
254 | if (rc < 0) throw MapWriteException(name); | ||
255 | |||
256 | // <environment/> | ||
257 | std::ostringstream mapdata_out; | ||
258 | for (int y=0; y<MAP_HEIGHT; y++) | ||
259 | { | ||
260 | for (int x=0; x<MAP_WIDTH; x++) | ||
261 | { | ||
262 | mapdata_out << map.getTileAt(x,y) << ","; | ||
263 | } | ||
264 | |||
265 | mapdata_out << std::endl; | ||
266 | } | ||
267 | |||
268 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "environment", (xmlChar*) mapdata_out.str().c_str()); | ||
269 | if (rc < 0) throw MapWriteException(name); | ||
270 | |||
271 | // <leftmap/> | ||
272 | std::ostringstream leftmap_out; | ||
273 | if (map.getLeftmap()) | ||
274 | { | ||
275 | leftmap_out << map.getLeftmap()->getID(); | ||
276 | } | ||
277 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "leftmap", (xmlChar*) leftmap_out.str().c_str()); | ||
278 | if (rc < 0) throw MapWriteException(name); | ||
279 | |||
280 | // <rightmap/> | ||
281 | std::ostringstream rightmap_out; | ||
282 | if (map.getRightmap()) | ||
283 | { | ||
284 | rightmap_out << map.getRightmap()->getID(); | ||
285 | } | ||
286 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "rightmap", (xmlChar*) rightmap_out.str().c_str()); | ||
287 | if (rc < 0) throw MapWriteException(name); | ||
288 | |||
289 | // <entities> | ||
290 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "entities"); | ||
291 | if (rc < 0) throw MapWriteException(name); | ||
292 | |||
293 | for (auto object : map.getObjects()) | ||
294 | { | ||
295 | // <entity> | ||
296 | rc = xmlTextWriterStartElement(writer, (xmlChar*) "entity"); | ||
297 | if (rc < 0) throw MapWriteException(name); | ||
298 | |||
299 | // <entity-type/> | ||
300 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "entity-type", (xmlChar*) object->object->getType().c_str()); | ||
301 | if (rc < 0) throw MapWriteException(name); | ||
302 | |||
303 | // <entity-position/> | ||
304 | std::ostringstream entpos_out; | ||
305 | entpos_out << object->position.first << "," << object->position.second; | ||
306 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "entity-position", (xmlChar*) entpos_out.str().c_str()); | ||
307 | if (rc < 0) throw MapWriteException(name); | ||
308 | |||
309 | // </entity> | ||
310 | rc = xmlTextWriterEndElement(writer); | ||
311 | if (rc < 0) throw MapWriteException(name); | ||
312 | } | ||
313 | |||
314 | // </entities> | ||
315 | rc = xmlTextWriterEndElement(writer); | ||
316 | if (rc < 0) throw MapWriteException(name); | ||
317 | |||
318 | wxTreeItemId node = map.getTreeItemId(); | ||
319 | if (mapTree->ItemHasChildren(node)) | ||
320 | { | ||
321 | wxTreeItemIdValue cookie2; | ||
322 | for (wxTreeItemId it = mapTree->GetFirstChild(node, cookie2); it.IsOk(); it = mapTree->GetNextChild(node, cookie2)) | ||
323 | { | ||
324 | // <child/> | ||
325 | MapPtrCtr* ctl = (MapPtrCtr*) mapTree->GetItemData(it); | ||
326 | std::ostringstream childid_out; | ||
327 | childid_out << ctl->map->getID(); | ||
328 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "child", (xmlChar*) childid_out.str().c_str()); | ||
329 | if (rc < 0) throw MapWriteException(name); | ||
330 | } | ||
331 | |||
332 | if (mapTree->IsExpanded(node)) | ||
333 | { | ||
334 | // <expanded/> | ||
335 | rc = xmlTextWriterWriteElement(writer, (xmlChar*) "expanded", (xmlChar*) "1"); | ||
336 | if (rc < 0) throw MapWriteException(name); | ||
337 | } | ||
338 | } | ||
339 | |||
340 | // </map> | ||
341 | rc = xmlTextWriterEndElement(writer); | ||
342 | if (rc < 0) throw MapWriteException(name); | ||
343 | } | ||
344 | |||
345 | // </world> | ||
346 | rc = xmlTextWriterEndDocument(writer); | ||
347 | if (rc < 0) throw MapWriteException(name); | ||
348 | |||
349 | xmlFreeTextWriter(writer); | ||
350 | |||
351 | setDirty(false); | ||
352 | } | ||
353 | |||
354 | std::list<std::shared_ptr<Map>> World::getRootMaps() const | ||
355 | { | ||
356 | std::list<std::shared_ptr<Map>> ret; | ||
357 | |||
358 | for (auto id : rootChildren) | ||
359 | { | ||
360 | ret.push_back(getMap(id)); | ||
361 | } | ||
362 | |||
363 | return ret; | ||
364 | } | ||
365 | |||
366 | const std::map<int, std::shared_ptr<Map>> World::getMaps() const | ||
367 | { | ||
368 | return maps; | ||
369 | } | ||
diff --git a/tools/mapedit/src/world.h b/tools/mapedit/src/world.h new file mode 100644 index 0000000..7d796cc --- /dev/null +++ b/tools/mapedit/src/world.h | |||
@@ -0,0 +1,45 @@ | |||
1 | #ifndef WORLD_H | ||
2 | #define WORLD_H | ||
3 | |||
4 | class World; | ||
5 | |||
6 | #include <wx/wxprec.h> | ||
7 | |||
8 | #ifndef WX_PRECOMP | ||
9 | #include <wx/wx.h> | ||
10 | #endif | ||
11 | |||
12 | #include <wx/treectrl.h> | ||
13 | #include <map> | ||
14 | #include <memory> | ||
15 | #include "map.h" | ||
16 | #include <list> | ||
17 | |||
18 | class MapeditFrame; | ||
19 | |||
20 | class World { | ||
21 | public: | ||
22 | World(); | ||
23 | World(std::string filename); | ||
24 | std::shared_ptr<Map> newMap(); | ||
25 | std::shared_ptr<Map> getMap(int id) const; | ||
26 | void setDirty(bool dirty); | ||
27 | bool getDirty() const; | ||
28 | std::string getFilename() const; | ||
29 | void setParent(MapeditFrame* parent); | ||
30 | void save(std::string filename, wxTreeCtrl* mapTree); | ||
31 | Map* getLastMap() const; | ||
32 | std::list<std::shared_ptr<Map>> getRootMaps() const; | ||
33 | const std::map<int, std::shared_ptr<Map>> getMaps() const; | ||
34 | |||
35 | private: | ||
36 | MapeditFrame* parent; | ||
37 | std::map<int, std::shared_ptr<Map>> maps; | ||
38 | int nextMapID = 0; | ||
39 | bool dirty = false; | ||
40 | std::string filename; | ||
41 | int lastmap = 0; | ||
42 | std::list<int> rootChildren; | ||
43 | }; | ||
44 | |||
45 | #endif | ||