1 /////////////////////////////////////////////////////////////////////////////
2 // Author: Vaclav Slavik
5 // Copyright: (c) 2000 Vaclav Slavik
6 // Licence: wxWindows licence
7 /////////////////////////////////////////////////////////////////////////////
10 #pragma implementation "editor.h"
11 #pragma implementation "treedt.h"
14 // For compilers that support precompilation, includes "wx/wx.h".
15 #include "wx/wxprec.h"
22 #include "wx/xrc/xml.h"
23 #include "wx/xrc/xmlres.h"
24 #include "wx/splitter.h"
25 #include "wx/config.h"
27 #include "wx/listctrl.h"
28 #include "wx/imaglist.h"
35 #include "propframe.h"
38 void wxXmlRcEditDocument::UpgradeNodeValue(wxXmlNode
*node
)
41 if (n
== NULL
) return;
46 if (n
->GetType() == wxXML_TEXT_NODE
||
47 n
->GetType() == wxXML_CDATA_SECTION_NODE
)
49 wxString str1
= n
->GetContent();
52 for (dt
= str1
.c_str(); *dt
; dt
++)
54 // Remap amp_char to &, map double amp_char to amp_char (for things
55 // like "&File..." -- this is illegal in XML, so we use "_File..."):
59 str1
[size_t(dt
-str1
.c_str()-1)] = '_';
68 void wxXmlRcEditDocument::UpgradeNode(wxXmlNode
*node
)
72 UpgradeNodeValue(node
);
73 UpgradeNode(node
->GetNext());
74 UpgradeNode(node
->GetChildren());
78 void wxXmlRcEditDocument::Upgrade()
82 wxXmlNode
*node
= GetRoot();
83 wxString verstr
= wxT("0.0.0.0");
84 node
->GetPropVal(wxT("version"),verstr
);
85 if (wxSscanf(verstr
.c_str(), wxT("%i.%i.%i.%i"),
86 &v1
, &v2
, &v3
, &v4
) == 4)
87 version
= v1
*256*256*256+v2
*256*256+v3
*256+v4
;
94 node
->DeleteProperty(wxT("version"));
95 node
->AddProperty(wxT("version"), WX_XMLRES_CURRENT_VERSION_STRING
);
99 class EditorTreeCtrl
: public wxTreeCtrl
102 EditorTreeCtrl(wxWindow
*parent
, int id
, EditorFrame
*frame
)
103 : wxTreeCtrl(parent
, id
), m_EdFrame(frame
) {}
106 EditorFrame
*m_EdFrame
;
108 void OnRightClick(wxMouseEvent
&event
)
111 m_EdFrame
->m_TreeCtrl
->HitTest(event
.GetPosition());
114 m_EdFrame
->m_TreeCtrl
->SelectItem(item
);
115 m_EdFrame
->OnRightClickTree(event
.GetPosition());
118 DECLARE_EVENT_TABLE()
121 BEGIN_EVENT_TABLE(EditorTreeCtrl
, wxTreeCtrl
)
122 EVT_RIGHT_DOWN(EditorTreeCtrl::OnRightClick
)
128 ID_PREVIEW
= wxID_HIGHEST
+ 100,
149 ID_NEWNODE
= wxID_HIGHEST
+ 10000, // safely out of XRCID range :)
150 ID_NEWSYBNODE
= ID_NEWNODE
+ 20000
157 BEGIN_EVENT_TABLE(EditorFrame
, wxFrame
)
158 EVT_TREE_SEL_CHANGED(ID_TREE
, EditorFrame::OnTreeSel
)
159 EVT_TOOL_RANGE(ID_PREVIEW
, ID_EXIT
, EditorFrame::OnToolbar
)
160 EVT_MENU_RANGE(ID_NEWDIALOG
, ID_NEWSYBNODE
+ 1000, EditorFrame::OnNewNode
)
161 EVT_MENU_RANGE(ID_CUT
, ID_COPY
, EditorFrame::OnClipboardAction
)
162 EVT_CLOSE(EditorFrame::OnCloseWindow
)
167 #if defined(__UNIX__)
168 #include "bitmaps/preview.xpm"
169 #include "bitmaps/close.xpm"
170 #include "bitmaps/save.xpm"
171 #include "bitmaps/open.xpm"
173 #include "bitmaps/control.xpm"
174 #include "bitmaps/vsizer.xpm"
175 #include "bitmaps/hsizer.xpm"
176 #include "bitmaps/panel.xpm"
177 #include "bitmaps/gsizer.xpm"
178 #include "bitmaps/resicon.xpm"
183 EditorFrame
*EditorFrame::ms_Instance
= NULL
;
185 EditorFrame::EditorFrame(wxFrame
*parent
, const wxString
& filename
)
186 : wxFrame(parent
, -1, filename
+ _("- wxWindows resources editor"))
193 wxConfigBase
*cfg
= wxConfigBase::Get();
195 SetSize(wxRect(wxPoint(cfg
->Read("editor_x", -1), cfg
->Read("editor_y", -1)),
196 wxSize(cfg
->Read("editor_w", 400), cfg
->Read("editor_h", 400))));
198 m_SelectedNode
= NULL
;
200 m_FileName
= wxEmptyString
;
202 wxMenu
*menuFile
= new wxMenu
;
203 menuFile
->Append(ID_NEW
, "&New");
204 menuFile
->Append(ID_OPEN
, "&Open\tCtrl-O");
205 menuFile
->Append(ID_SAVE
, "&Save\tCtrl-S");
206 menuFile
->Append(ID_SAVEAS
, "Save &as...");
207 menuFile
->AppendSeparator();
208 menuFile
->Append(ID_EXIT
, "E&xit\tAlt-X");
210 wxMenu
*menuEdit
= new wxMenu
;
211 menuEdit
->Append(ID_CUT
, "Cut\tCtrl-X");
212 menuEdit
->Append(ID_COPY
, "Copy\tCtrl-C");
213 menuEdit
->Append(ID_PASTE_SYBLING
, "Paste as sybling\tCtrl-V");
214 menuEdit
->Append(ID_PASTE_CHILD
, "Paste as child");
215 menuEdit
->AppendSeparator();
216 menuEdit
->Append(ID_DELETE_NODE
, "Delete");
218 menuEdit
->Enable(ID_PASTE_SYBLING
, FALSE
);
219 menuEdit
->Enable(ID_PASTE_CHILD
, FALSE
);
221 wxMenuBar
*menuBar
= new wxMenuBar();
222 menuBar
->Append(menuFile
, "&File");
223 menuBar
->Append(menuEdit
, "&Edit");
227 wxToolBar
*toolBar
= CreateToolBar(wxNO_BORDER
| wxTB_HORIZONTAL
| wxTB_FLAT
);
228 toolBar
->SetMargins(2, 2);
229 toolBar
->SetToolBitmapSize(wxSize(24, 24));
230 toolBar
-> AddTool(ID_EXIT
, wxBITMAP(close
), wxNullBitmap
,
231 FALSE
, -1, -1, (wxObject
*) NULL
,
232 _("Quit the editor"));
233 toolBar
-> AddTool(ID_OPEN
, wxBITMAP(open
), wxNullBitmap
,
234 FALSE
, -1, -1, (wxObject
*) NULL
,
235 _("Open XML resource file"));
236 toolBar
-> AddTool(ID_SAVE
, wxBITMAP(save
), wxNullBitmap
,
237 FALSE
, -1, -1, (wxObject
*) NULL
,
239 toolBar
-> AddTool(ID_PREVIEW
, wxBITMAP(preview
), wxNullBitmap
,
240 FALSE
, -1, -1, (wxObject
*) NULL
,
242 toolBar
-> Realize();
244 wxSizer
*sizer
= new wxBoxSizer(wxHORIZONTAL
);
246 // Create tree control:
247 m_TreeCtrl
= new EditorTreeCtrl(this, ID_TREE
, this);
248 wxImageList
*imgList
= new wxImageList(16, 16);
249 imgList
->Add(wxICON(control
));
250 imgList
->Add(wxICON(panel
));
251 imgList
->Add(wxICON(vsizer
));
252 imgList
->Add(wxICON(hsizer
));
253 imgList
->Add(wxICON(gsizer
));
254 imgList
->Add(wxICON(resicon
));
255 m_TreeCtrl
->AssignImageList(imgList
);
256 sizer
->Add(m_TreeCtrl
, 1, wxEXPAND
);
270 EditorFrame::~EditorFrame()
272 PreviewFrame::Get()->Close();
273 PropertiesFrame::Get()->Close();
275 wxConfigBase
*cfg
= wxConfigBase::Get();
277 cfg
->Write(_T("editor_x"), (long)GetPosition().x
);
278 cfg
->Write(_T("editor_y"), (long)GetPosition().y
);
279 cfg
->Write(_T("editor_w"), (long)GetSize().x
);
280 cfg
->Write(_T("editor_h"), (long)GetSize().y
);
288 void EditorFrame::LoadFile(const wxString
& filename
)
290 if (!AskToSave()) return;
294 // create new resource in order to handle version differences properly
295 PreviewFrame::Get()->ResetResource();
298 m_Resource
= new wxXmlRcEditDocument
;
301 if (!m_Resource
->Load(filename
, wxLocale::GetSystemEncodingName()))
306 wxLogError("Error parsing " + filename
);
310 m_FileName
= filename
;
312 // Upgrades old versions
313 m_Resource
->Upgrade();
321 void EditorFrame::SaveFile(const wxString
& filename
)
323 m_FileName
= filename
;
326 if (!m_Resource
->Save(filename
))
327 wxLogError(_("Error saving ") + filename
);
336 void EditorFrame::NewFile()
338 if (!AskToSave()) return;
343 m_Resource
= new wxXmlRcEditDocument
;
344 m_Resource
->SetRoot(new wxXmlNode(wxXML_ELEMENT_NODE
, _("resource")));
346 m_Resource
->SetFileEncoding("utf-8");
348 m_Resource
->GetRoot()->AddProperty(_T("version"),
349 WX_XMLRES_CURRENT_VERSION_STRING
);
358 void EditorFrame::RefreshTitle()
361 if (m_Modified
) s
<< _T("* ");
363 if (m_FileName
!= "")
364 s
<< _T(" - ") << wxFileNameFromPath(m_FileName
);
370 void EditorFrame::RefreshTree()
372 wxXmlNode
*sel
= m_SelectedNode
;
374 m_TreeCtrl
->DeleteAllItems();
376 wxTreeItemId root
= m_TreeCtrl
->AddRoot("Resource: " + wxFileNameFromPath(m_FileName
), 5, 5);
378 wxXmlNode
*n
= m_Resource
->GetRoot()->GetChildren();
381 if (n
->GetType() == wxXML_ELEMENT_NODE
)
382 CreateTreeNode(m_TreeCtrl
, root
, n
);
386 m_TreeCtrl
->Expand(root
);
393 static void RecursivelyExpand(wxTreeCtrl
*t
, wxTreeItemId item
)
397 wxTreeItemId id
= t
->GetFirstChild(item
, cookie
);
400 RecursivelyExpand(t
, id
);
401 id
= t
->GetNextChild(item
, cookie
);
405 bool EditorFrame::SelectNode(wxXmlNode
*node
, wxTreeItemId
*root
)
409 wxTreeItemId rootitem
= m_TreeCtrl
->GetRootItem();
410 return SelectNode(node
, &rootitem
);
418 item
= m_TreeCtrl
->GetFirstChild(*root
, cookie
);
421 dt
= (XmlTreeData
*)(m_TreeCtrl
->GetItemData(item
));
422 nd
= (dt
) ? dt
->Node
: NULL
;
425 RecursivelyExpand(m_TreeCtrl
, *root
);
426 m_TreeCtrl
->SelectItem(item
);
427 m_TreeCtrl
->EnsureVisible(item
);
430 if (m_TreeCtrl
->ItemHasChildren(item
) && SelectNode(node
, &item
))
432 item
= m_TreeCtrl
->GetNextChild(*root
, cookie
);
440 wxTreeItemId
EditorFrame::CreateTreeNode(wxTreeCtrl
*treectrl
, wxTreeItemId parent
, wxXmlNode
*node
)
444 wxTreeItemId invalid
;
448 return NodeHandler::Find(node
)->CreateTreeNode(treectrl
, parent
, node
);
453 void EditorFrame::NotifyChanged(int change_type
)
455 if (change_type
& CHANGED_TREE
)
458 if (change_type
& CHANGED_TREE_SELECTED
)
460 wxTreeItemId sel
= m_TreeCtrl
->GetSelection();
461 m_TreeCtrl
->SetItemText(sel
,
462 NodeHandler::Find(m_SelectedNode
)->GetTreeString(m_SelectedNode
));
465 if (change_type
& CHANGED_TREE_SELECTED_ICON
)
467 wxTreeItemId sel
= m_TreeCtrl
->GetSelection();
468 int icon
= NodeHandler::Find(m_SelectedNode
)->GetTreeIcon(m_SelectedNode
);
469 m_TreeCtrl
->SetItemImage(sel
, icon
);
478 PreviewFrame::Get()->MakeDirty();
483 void EditorFrame::OnTreeSel(wxTreeEvent
& event
)
485 XmlTreeData
*dt
= (XmlTreeData
*)(m_TreeCtrl
->GetItemData(event
.GetItem()));
486 wxXmlNode
*node
= (dt
) ? dt
->Node
: NULL
;
488 m_SelectedNode
= node
;
490 PropertiesFrame::Get()->ShowProps(node
);
492 if (m_TreeCtrl
->GetParent(event
.GetItem()) == m_TreeCtrl
->GetRootItem())
494 wxTreeItemId it
= event
.GetOldItem();
496 if (it
.IsOk() && m_TreeCtrl
->GetRootItem() != it
)
498 while (m_TreeCtrl
->GetParent(it
) != m_TreeCtrl
->GetRootItem())
499 it
= m_TreeCtrl
->GetParent(it
);
500 m_TreeCtrl
->Collapse(it
);
502 RecursivelyExpand(m_TreeCtrl
, event
.GetItem());
504 PreviewFrame::Get()->Preview(node
,m_Resource
);
510 void EditorFrame::OnToolbar(wxCommandEvent
& event
)
512 switch (event
.GetId())
516 XmlTreeData
* dt
= (XmlTreeData
*)m_TreeCtrl
->GetItemData(m_TreeCtrl
->GetSelection());;
517 if (dt
!= NULL
&& dt
->Node
!= NULL
)
518 PreviewFrame::Get()->Preview(dt
->Node
, m_Resource
);
532 wxString cwd
= wxGetCwd(); // workaround for 2.2
533 wxString name
= wxFileSelector(_("Open XML resource"), _T(""), _T(""), _T(""), _("XML resources (*.xrc)|*.xrc"), wxOPEN
| wxFILE_MUST_EXIST
);
534 wxSetWorkingDirectory(cwd
);
541 if (m_FileName
!= "") { SaveFile(m_FileName
); break;}
546 wxString cwd
= wxGetCwd(); // workaround for 2.2
547 wxString name
= wxFileSelector(_("Save as"), _T(""), m_FileName
, _T(""), _("XML resources (*.xrc)|*.xrc"), wxSAVE
| wxOVERWRITE_PROMPT
);
548 wxSetWorkingDirectory(cwd
);
550 SaveFile((m_FileName
= name
));
554 case ID_DELETE_NODE
:
556 DeleteSelectedNode();
564 void EditorFrame::DeleteSelectedNode()
566 XmlTreeData
*dt
= (XmlTreeData
*)
567 (m_TreeCtrl
->GetItemData(m_TreeCtrl
->GetParent(m_TreeCtrl
->GetSelection())));
568 wxXmlNode
*n
= (dt
) ? dt
->Node
: NULL
;
570 m_SelectedNode
->GetParent()->RemoveChild(m_SelectedNode
);
571 NotifyChanged(CHANGED_TREE
);
577 void EditorFrame::OnNewNode(wxCommandEvent
& event
)
579 if (event
.GetId() >= ID_NEWSYBNODE
)
582 (XmlTreeData
*)(m_TreeCtrl
->GetItemData(
583 m_TreeCtrl
->GetParent(m_TreeCtrl
->GetSelection())));
585 if (pardt
&& pardt
->Node
&& pardt
->Node
!= m_Resource
->GetRoot())
587 wxXmlNode
*nd
= pardt
->Node
;
589 wxXmlNode
*realnode
= NodeHandler::Find(nd
)->GetRealNode(nd
);
590 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
591 wxString name
= hnd
->GetChildTypes()[event
.GetId()-ID_NEWSYBNODE
];
593 wxXmlNode
*node
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("object"));
594 node
->AddProperty(_T("class"), name
);
596 hnd
->InsertNode(realnode
, node
, m_SelectedNode
);
597 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
598 SelectNode(node
, &root
);
603 else if (event
.GetId() >= ID_NEWNODE
)
605 wxXmlNode
*realnode
= NodeHandler::Find(m_SelectedNode
)->GetRealNode(m_SelectedNode
);
606 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
607 wxString name
= hnd
->GetChildTypes()[event
.GetId()-ID_NEWNODE
];
609 wxXmlNode
*node
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("object"));
610 node
->AddProperty(_T("class"), name
);
612 hnd
->InsertNode(realnode
, node
);
613 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
614 SelectNode(node
, &root
);
620 switch (event
.GetId())
622 case ID_NEWDIALOG
: name
= _T("wxDialog"); break;
623 case ID_NEWFRAME
: name
= _T("wxFrame"); break;
624 case ID_NEWPANEL
: name
= _T("wxPanel"); break;
625 case ID_NEWMENU
: name
= _T("wxMenu"); break;
626 case ID_NEWMENUBAR
: name
= _T("wxMenuBar"); break;
627 case ID_NEWTOOLBAR
: name
= _T("wxToolBar"); break;
628 default : return; // never occurs
631 wxXmlNode
*node
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("object"));
632 node
->AddProperty(_T("class"), name
);
633 m_Resource
->GetRoot()->AddChild(node
);
634 NotifyChanged(CHANGED_TREE
);
641 void EditorFrame::OnRightClickTree(wxPoint pos
)
643 wxMenu
*popup
= new wxMenu
;
645 if (m_SelectedNode
== NULL
|| m_SelectedNode
== m_Resource
->GetRoot())
647 popup
->Append(ID_NEWDIALOG
, _("New wxDialog"));
648 popup
->Append(ID_NEWFRAME
, _("New wxFrame"));
649 popup
->Append(ID_NEWPANEL
, _("New wxPanel"));
650 popup
->Append(ID_NEWMENU
, _("New wxMenu"));
651 popup
->Append(ID_NEWMENUBAR
, _("New wxMenuBar"));
652 popup
->Append(ID_NEWTOOLBAR
, _("New wxToolBar"));
660 NodeHandler::Find(NodeHandler::Find(m_SelectedNode
)->GetRealNode(m_SelectedNode
))->
663 has_children
= !arr
.IsEmpty();
666 wxMenu
*news
= new wxMenu
;
667 wxMenu
*news2
= news
;
668 for (size_t i
= 0; i
< arr
.GetCount(); i
++)
670 news2
->Append(i
+ ID_NEWNODE
, arr
[i
]);
671 #ifdef __WXGTK__ // doesn't support Break
674 wxMenu
*m
= new wxMenu
;
675 news2
->Append(ID_NEWNODE
+arr
.GetCount(), _("More..."), m
);
679 if (i
% 16 == 15) news2
->Break();
682 popup
->Append(ID_NEWNODE
-1, _("New child"), news
);
688 (XmlTreeData
*)(m_TreeCtrl
->GetItemData(
689 m_TreeCtrl
->GetParent(m_TreeCtrl
->GetSelection())));
690 if (pardt
&& pardt
->Node
&& pardt
->Node
!= m_Resource
->GetRoot())
692 wxXmlNode
*nd
= pardt
->Node
;
694 NodeHandler::Find(NodeHandler::Find(nd
)->GetRealNode(nd
))->
699 wxMenu
*news
= new wxMenu
;
700 wxMenu
*news2
= news
;
701 for (size_t i
= 0; i
< arr
.GetCount(); i
++)
703 news2
->Append(i
+ ID_NEWSYBNODE
, arr
[i
]);
704 #ifdef __WXGTK__ // doesn't support Break
707 wxMenu
*m
= new wxMenu
;
708 news2
->Append(ID_NEWSYBNODE
+arr
.GetCount(), _("More..."), m
);
712 if (i
% 16 == 15) news2
->Break();
715 popup
->Append(ID_NEWSYBNODE
-1, _("New sybling"), news
);
720 popup
->AppendSeparator();
721 popup
->Append(ID_CUT
, _("Cut"));
722 popup
->Append(ID_COPY
, _("Copy"));
723 popup
->Append(ID_PASTE_SYBLING
, _("Paste as sybling"));
724 popup
->Append(ID_PASTE_CHILD
, _("Paste as child"));
725 popup
->AppendSeparator();
726 popup
->Append(ID_DELETE_NODE
, _("Delete"));
727 popup
->Enable(ID_PASTE_SYBLING
, m_Clipboard
!= NULL
);
728 popup
->Enable(ID_PASTE_CHILD
, has_children
&& m_Clipboard
!= NULL
);
731 m_TreeCtrl
->PopupMenu(popup
, pos
);
737 void EditorFrame::OnClipboardAction(wxCommandEvent
& event
)
739 switch (event
.GetId())
744 m_Clipboard
= new wxXmlNode(*m_SelectedNode
);
745 GetMenuBar()->Enable(ID_PASTE_SYBLING
, TRUE
);
746 GetMenuBar()->Enable(ID_PASTE_CHILD
, TRUE
);
747 if (event
.GetId() == ID_CUT
) DeleteSelectedNode();
750 case ID_PASTE_SYBLING
:
753 (XmlTreeData
*)(m_TreeCtrl
->GetItemData(
754 m_TreeCtrl
->GetParent(m_TreeCtrl
->GetSelection())));
756 if (pardt
&& pardt
->Node
&& pardt
->Node
!= m_Resource
->GetRoot())
758 wxXmlNode
*nd
= pardt
->Node
;
760 wxXmlNode
*realnode
= NodeHandler::Find(nd
)->GetRealNode(nd
);
761 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
762 wxXmlNode
*node
= new wxXmlNode(*m_Clipboard
);
763 hnd
->InsertNode(realnode
, node
, m_SelectedNode
);
764 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
765 SelectNode(node
, &root
);
771 wxXmlNode
*realnode
= NodeHandler::Find(m_SelectedNode
)->GetRealNode(m_SelectedNode
);
772 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
773 wxXmlNode
*node
= new wxXmlNode(*m_Clipboard
);
774 hnd
->InsertNode(realnode
, node
);
775 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
776 SelectNode(node
, &root
);
784 bool EditorFrame::AskToSave()
785 // asks the user to save current document (if modified)
786 // returns FALSE if user cancelled the action, TRUE of he choosed
789 if (!m_Modified
) return TRUE
;
791 int res
= wxMessageBox(_("File modified. Do you want to save changes?"), _("Save changes"),
792 wxYES_NO
| wxCANCEL
| wxCENTRE
| wxICON_QUESTION
);
794 SaveFile(m_FileName
);
795 return (res
!= wxCANCEL
);
800 void EditorFrame::OnCloseWindow(wxCloseEvent
&)
802 if (!AskToSave()) return;