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/xml/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
+ _("- wxWidgets resources editor"))
193 wxConfigBase
*cfg
= wxConfigBase::Get();
195 SetSize(wxRect(wxPoint(cfg
->Read(_T("editor_x"), -1), cfg
->Read(_T("editor_y"), -1)),
196 wxSize(cfg
->Read(_T("editor_w"), 400), cfg
->Read(_T("editor_h"), 400))));
198 m_SelectedNode
= NULL
;
200 m_FileName
= wxEmptyString
;
202 wxMenu
*menuFile
= new wxMenu
;
203 menuFile
->Append(ID_NEW
, _T("&New"));
204 menuFile
->Append(ID_OPEN
, _T("&Open\tCtrl-O"));
205 menuFile
->Append(ID_SAVE
, _T("&Save\tCtrl-S"));
206 menuFile
->Append(ID_SAVEAS
, _T("Save &as..."));
207 menuFile
->AppendSeparator();
208 menuFile
->Append(ID_EXIT
, _T("E&xit\tAlt-X"));
210 wxMenu
*menuEdit
= new wxMenu
;
211 menuEdit
->Append(ID_CUT
, _T("Cut\tCtrl-X"));
212 menuEdit
->Append(ID_COPY
, _T("Copy\tCtrl-C"));
213 menuEdit
->Append(ID_PASTE_SYBLING
, _T("Paste as sybling\tCtrl-V"));
214 menuEdit
->Append(ID_PASTE_CHILD
, _T("Paste as child"));
215 menuEdit
->AppendSeparator();
216 menuEdit
->Append(ID_DELETE_NODE
, _T("Delete"));
218 menuEdit
->Enable(ID_PASTE_SYBLING
, FALSE
);
219 menuEdit
->Enable(ID_PASTE_CHILD
, FALSE
);
221 wxMenuBar
*menuBar
= new wxMenuBar();
222 menuBar
->Append(menuFile
, _T("&File"));
223 menuBar
->Append(menuEdit
, _T("&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();
297 m_FileName
= wxEmptyString
;
298 m_Resource
= new wxXmlRcEditDocument
;
301 if (!m_Resource
->Load(filename
, wxLocale::GetSystemEncodingName()))
306 wxLogError(_T("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;
342 m_FileName
= wxEmptyString
;
343 m_Resource
= new wxXmlRcEditDocument
;
344 m_Resource
->SetRoot(new wxXmlNode(wxXML_ELEMENT_NODE
, _("resource")));
346 m_Resource
->SetFileEncoding(_T("utf-8"));
348 m_Resource
->SetEncoding(wxLocale::GetSystemEncodingName());
351 m_Resource
->GetRoot()->AddProperty(_T("version"),
352 WX_XMLRES_CURRENT_VERSION_STRING
);
361 void EditorFrame::RefreshTitle()
364 if (m_Modified
) s
<< _T("* ");
366 if (m_FileName
!= wxEmptyString
)
367 s
<< _T(" - ") << wxFileNameFromPath(m_FileName
);
373 void EditorFrame::RefreshTree()
375 wxXmlNode
*sel
= m_SelectedNode
;
377 m_TreeCtrl
->DeleteAllItems();
379 wxTreeItemId root
= m_TreeCtrl
->AddRoot(_T("Resource: ") + wxFileNameFromPath(m_FileName
), 5, 5);
381 wxXmlNode
*n
= m_Resource
->GetRoot()->GetChildren();
384 if (n
->GetType() == wxXML_ELEMENT_NODE
)
385 CreateTreeNode(m_TreeCtrl
, root
, n
);
389 m_TreeCtrl
->Expand(root
);
396 static void RecursivelyExpand(wxTreeCtrl
*t
, wxTreeItemId item
)
400 wxTreeItemId id
= t
->GetFirstChild(item
, cookie
);
403 RecursivelyExpand(t
, id
);
404 id
= t
->GetNextChild(item
, cookie
);
408 bool EditorFrame::SelectNode(wxXmlNode
*node
, wxTreeItemId
*root
)
412 wxTreeItemId rootitem
= m_TreeCtrl
->GetRootItem();
413 return SelectNode(node
, &rootitem
);
421 item
= m_TreeCtrl
->GetFirstChild(*root
, cookie
);
424 dt
= (XmlTreeData
*)(m_TreeCtrl
->GetItemData(item
));
425 nd
= (dt
) ? dt
->Node
: NULL
;
428 RecursivelyExpand(m_TreeCtrl
, *root
);
429 m_TreeCtrl
->SelectItem(item
);
430 m_TreeCtrl
->EnsureVisible(item
);
433 if (m_TreeCtrl
->ItemHasChildren(item
) && SelectNode(node
, &item
))
435 item
= m_TreeCtrl
->GetNextChild(*root
, cookie
);
443 wxTreeItemId
EditorFrame::CreateTreeNode(wxTreeCtrl
*treectrl
, wxTreeItemId parent
, wxXmlNode
*node
)
447 wxTreeItemId invalid
;
451 return NodeHandler::Find(node
)->CreateTreeNode(treectrl
, parent
, node
);
456 void EditorFrame::NotifyChanged(int change_type
)
458 if (change_type
& CHANGED_TREE
)
461 if (change_type
& CHANGED_TREE_SELECTED
)
463 wxTreeItemId sel
= m_TreeCtrl
->GetSelection();
464 m_TreeCtrl
->SetItemText(sel
,
465 NodeHandler::Find(m_SelectedNode
)->GetTreeString(m_SelectedNode
));
468 if (change_type
& CHANGED_TREE_SELECTED_ICON
)
470 wxTreeItemId sel
= m_TreeCtrl
->GetSelection();
471 int icon
= NodeHandler::Find(m_SelectedNode
)->GetTreeIcon(m_SelectedNode
);
472 m_TreeCtrl
->SetItemImage(sel
, icon
);
481 PreviewFrame::Get()->MakeDirty();
486 void EditorFrame::OnTreeSel(wxTreeEvent
& event
)
488 XmlTreeData
*dt
= (XmlTreeData
*)(m_TreeCtrl
->GetItemData(event
.GetItem()));
489 wxXmlNode
*node
= (dt
) ? dt
->Node
: NULL
;
491 m_SelectedNode
= node
;
493 PropertiesFrame::Get()->ShowProps(node
);
495 if (m_TreeCtrl
->GetItemParent(event
.GetItem()) == m_TreeCtrl
->GetRootItem())
497 wxTreeItemId it
= event
.GetOldItem();
499 if (it
.IsOk() && m_TreeCtrl
->GetRootItem() != it
)
501 while (m_TreeCtrl
->GetItemParent(it
) != m_TreeCtrl
->GetRootItem())
502 it
= m_TreeCtrl
->GetItemParent(it
);
503 m_TreeCtrl
->Collapse(it
);
505 RecursivelyExpand(m_TreeCtrl
, event
.GetItem());
507 PreviewFrame::Get()->Preview(node
,m_Resource
);
513 void EditorFrame::OnToolbar(wxCommandEvent
& event
)
515 switch (event
.GetId())
519 XmlTreeData
* dt
= (XmlTreeData
*)m_TreeCtrl
->GetItemData(m_TreeCtrl
->GetSelection());;
520 if (dt
!= NULL
&& dt
->Node
!= NULL
)
521 PreviewFrame::Get()->Preview(dt
->Node
, m_Resource
);
535 wxString cwd
= wxGetCwd(); // workaround for 2.2
536 wxString name
= wxFileSelector(_("Open XML resource"), _T(""), _T(""), _T(""), _("XML resources (*.xrc)|*.xrc"), wxOPEN
| wxFILE_MUST_EXIST
);
537 wxSetWorkingDirectory(cwd
);
544 if (m_FileName
!= wxEmptyString
) { SaveFile(m_FileName
); break;}
549 wxString cwd
= wxGetCwd(); // workaround for 2.2
550 wxString name
= wxFileSelector(_("Save as"), _T(""), m_FileName
, _T(""), _("XML resources (*.xrc)|*.xrc"), wxSAVE
| wxOVERWRITE_PROMPT
);
551 wxSetWorkingDirectory(cwd
);
553 SaveFile((m_FileName
= name
));
557 case ID_DELETE_NODE
:
559 DeleteSelectedNode();
567 void EditorFrame::DeleteSelectedNode()
569 XmlTreeData
*dt
= (XmlTreeData
*)
570 (m_TreeCtrl
->GetItemData(m_TreeCtrl
->GetItemParent(m_TreeCtrl
->GetSelection())));
571 wxXmlNode
*n
= (dt
) ? dt
->Node
: NULL
;
573 m_SelectedNode
->GetParent()->RemoveChild(m_SelectedNode
);
574 NotifyChanged(CHANGED_TREE
);
580 void EditorFrame::OnNewNode(wxCommandEvent
& event
)
582 if (event
.GetId() >= ID_NEWSYBNODE
)
585 (XmlTreeData
*)(m_TreeCtrl
->GetItemData(
586 m_TreeCtrl
->GetItemParent(m_TreeCtrl
->GetSelection())));
588 if (pardt
&& pardt
->Node
&& pardt
->Node
!= m_Resource
->GetRoot())
590 wxXmlNode
*nd
= pardt
->Node
;
592 wxXmlNode
*realnode
= NodeHandler::Find(nd
)->GetRealNode(nd
);
593 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
594 wxString name
= hnd
->GetChildTypes()[event
.GetId()-ID_NEWSYBNODE
];
596 wxXmlNode
*node
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("object"));
597 node
->AddProperty(_T("class"), name
);
599 hnd
->InsertNode(realnode
, node
, m_SelectedNode
);
600 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
601 SelectNode(node
, &root
);
606 else if (event
.GetId() >= ID_NEWNODE
)
608 wxXmlNode
*realnode
= NodeHandler::Find(m_SelectedNode
)->GetRealNode(m_SelectedNode
);
609 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
610 wxString name
= hnd
->GetChildTypes()[event
.GetId()-ID_NEWNODE
];
612 wxXmlNode
*node
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("object"));
613 node
->AddProperty(_T("class"), name
);
615 hnd
->InsertNode(realnode
, node
);
616 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
617 SelectNode(node
, &root
);
623 switch (event
.GetId())
625 case ID_NEWDIALOG
: name
= _T("wxDialog"); break;
626 case ID_NEWFRAME
: name
= _T("wxFrame"); break;
627 case ID_NEWPANEL
: name
= _T("wxPanel"); break;
628 case ID_NEWMENU
: name
= _T("wxMenu"); break;
629 case ID_NEWMENUBAR
: name
= _T("wxMenuBar"); break;
630 case ID_NEWTOOLBAR
: name
= _T("wxToolBar"); break;
631 default : return; // never occurs
634 wxXmlNode
*node
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("object"));
635 node
->AddProperty(_T("class"), name
);
636 m_Resource
->GetRoot()->AddChild(node
);
637 NotifyChanged(CHANGED_TREE
);
644 void EditorFrame::OnRightClickTree(wxPoint pos
)
646 wxMenu
*popup
= new wxMenu
;
648 if (m_SelectedNode
== NULL
|| m_SelectedNode
== m_Resource
->GetRoot())
650 popup
->Append(ID_NEWDIALOG
, _("New wxDialog"));
651 popup
->Append(ID_NEWFRAME
, _("New wxFrame"));
652 popup
->Append(ID_NEWPANEL
, _("New wxPanel"));
653 popup
->Append(ID_NEWMENU
, _("New wxMenu"));
654 popup
->Append(ID_NEWMENUBAR
, _("New wxMenuBar"));
655 popup
->Append(ID_NEWTOOLBAR
, _("New wxToolBar"));
663 NodeHandler::Find(NodeHandler::Find(m_SelectedNode
)->GetRealNode(m_SelectedNode
))->
666 has_children
= !arr
.IsEmpty();
669 wxMenu
*news
= new wxMenu
;
670 wxMenu
*news2
= news
;
671 for (size_t i
= 0; i
< arr
.GetCount(); i
++)
673 news2
->Append(i
+ ID_NEWNODE
, arr
[i
]);
674 #ifdef __WXGTK__ // doesn't support Break
677 wxMenu
*m
= new wxMenu
;
678 news2
->Append(ID_NEWNODE
+arr
.GetCount(), _("More..."), m
);
682 if (i
% 16 == 15) news2
->Break();
685 popup
->Append(ID_NEWNODE
-1, _("New child"), news
);
691 (XmlTreeData
*)(m_TreeCtrl
->GetItemData(
692 m_TreeCtrl
->GetItemParent(m_TreeCtrl
->GetSelection())));
693 if (pardt
&& pardt
->Node
&& pardt
->Node
!= m_Resource
->GetRoot())
695 wxXmlNode
*nd
= pardt
->Node
;
697 NodeHandler::Find(NodeHandler::Find(nd
)->GetRealNode(nd
))->
702 wxMenu
*news
= new wxMenu
;
703 wxMenu
*news2
= news
;
704 for (size_t i
= 0; i
< arr
.GetCount(); i
++)
706 news2
->Append(i
+ ID_NEWSYBNODE
, arr
[i
]);
707 #ifdef __WXGTK__ // doesn't support Break
710 wxMenu
*m
= new wxMenu
;
711 news2
->Append(ID_NEWSYBNODE
+arr
.GetCount(), _("More..."), m
);
715 if (i
% 16 == 15) news2
->Break();
718 popup
->Append(ID_NEWSYBNODE
-1, _("New sybling"), news
);
723 popup
->AppendSeparator();
724 popup
->Append(ID_CUT
, _("Cut"));
725 popup
->Append(ID_COPY
, _("Copy"));
726 popup
->Append(ID_PASTE_SYBLING
, _("Paste as sybling"));
727 popup
->Append(ID_PASTE_CHILD
, _("Paste as child"));
728 popup
->AppendSeparator();
729 popup
->Append(ID_DELETE_NODE
, _("Delete"));
730 popup
->Enable(ID_PASTE_SYBLING
, m_Clipboard
!= NULL
);
731 popup
->Enable(ID_PASTE_CHILD
, has_children
&& m_Clipboard
!= NULL
);
734 m_TreeCtrl
->PopupMenu(popup
, pos
);
740 void EditorFrame::OnClipboardAction(wxCommandEvent
& event
)
742 switch (event
.GetId())
747 m_Clipboard
= new wxXmlNode(*m_SelectedNode
);
748 GetMenuBar()->Enable(ID_PASTE_SYBLING
, TRUE
);
749 GetMenuBar()->Enable(ID_PASTE_CHILD
, TRUE
);
750 if (event
.GetId() == ID_CUT
) DeleteSelectedNode();
753 case ID_PASTE_SYBLING
:
756 (XmlTreeData
*)(m_TreeCtrl
->GetItemData(
757 m_TreeCtrl
->GetItemParent(m_TreeCtrl
->GetSelection())));
759 if (pardt
&& pardt
->Node
&& pardt
->Node
!= m_Resource
->GetRoot())
761 wxXmlNode
*nd
= pardt
->Node
;
763 wxXmlNode
*realnode
= NodeHandler::Find(nd
)->GetRealNode(nd
);
764 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
765 wxXmlNode
*node
= new wxXmlNode(*m_Clipboard
);
766 hnd
->InsertNode(realnode
, node
, m_SelectedNode
);
767 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
768 SelectNode(node
, &root
);
774 wxXmlNode
*realnode
= NodeHandler::Find(m_SelectedNode
)->GetRealNode(m_SelectedNode
);
775 NodeHandler
*hnd
= NodeHandler::Find(realnode
);
776 wxXmlNode
*node
= new wxXmlNode(*m_Clipboard
);
777 hnd
->InsertNode(realnode
, node
);
778 wxTreeItemId root
= m_TreeCtrl
->GetSelection();
779 SelectNode(node
, &root
);
787 bool EditorFrame::AskToSave()
788 // asks the user to save current document (if modified)
789 // returns FALSE if user cancelled the action, TRUE of he choosed
792 if (!m_Modified
) return TRUE
;
794 int res
= wxMessageBox(_("File modified. Do you want to save changes?"), _("Save changes"),
795 wxYES_NO
| wxCANCEL
| wxCENTRE
| wxICON_QUESTION
);
797 SaveFile(m_FileName
);
798 return (res
!= wxCANCEL
);
803 void EditorFrame::OnCloseWindow(wxCloseEvent
&)
805 if (!AskToSave()) return;