2000-07-25 18:47:21 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Author: Vaclav Slavik
|
|
|
|
// Created: 2000/05/05
|
|
|
|
// RCS-ID: $Id$
|
|
|
|
// Copyright: (c) 2000 Vaclav Slavik
|
|
|
|
// Licence: wxWindows licence
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#ifdef __GNUG__
|
|
|
|
#pragma implementation "editor.h"
|
|
|
|
#pragma implementation "treedt.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// For compilers that support precompilation, includes "wx/wx.h".
|
|
|
|
#include "wx/wxprec.h"
|
|
|
|
|
|
|
|
#ifdef __BORLANDC__
|
|
|
|
#pragma hdrstop
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "wx/wx.h"
|
2003-09-10 20:53:01 +00:00
|
|
|
#include "wx/xml/xml.h"
|
2001-06-09 22:18:45 +00:00
|
|
|
#include "wx/xrc/xmlres.h"
|
2000-07-25 18:47:21 +00:00
|
|
|
#include "wx/splitter.h"
|
|
|
|
#include "wx/config.h"
|
|
|
|
#include "wx/dir.h"
|
2000-07-26 21:49:18 +00:00
|
|
|
#include "wx/listctrl.h"
|
|
|
|
#include "wx/imaglist.h"
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
#include "treedt.h"
|
|
|
|
#include "editor.h"
|
|
|
|
#include "nodehnd.h"
|
|
|
|
#include "xmlhelpr.h"
|
2000-10-07 21:56:37 +00:00
|
|
|
#include "preview.h"
|
2000-11-04 23:34:33 +00:00
|
|
|
#include "propframe.h"
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
|
2001-05-29 18:21:09 +00:00
|
|
|
void wxXmlRcEditDocument::UpgradeNodeValue(wxXmlNode *node)
|
|
|
|
{
|
|
|
|
wxXmlNode *n = node;
|
|
|
|
if (n == NULL) return;
|
|
|
|
n = n->GetChildren();
|
|
|
|
|
|
|
|
while (n)
|
|
|
|
{
|
|
|
|
if (n->GetType() == wxXML_TEXT_NODE ||
|
|
|
|
n->GetType() == wxXML_CDATA_SECTION_NODE)
|
|
|
|
{
|
|
|
|
wxString str1 = n->GetContent();
|
|
|
|
const wxChar *dt;
|
|
|
|
|
|
|
|
for (dt = str1.c_str(); *dt; dt++)
|
|
|
|
{
|
|
|
|
// Remap amp_char to &, map double amp_char to amp_char (for things
|
|
|
|
// like "&File..." -- this is illegal in XML, so we use "_File..."):
|
|
|
|
if (*dt == '$')
|
|
|
|
{
|
|
|
|
if ( *(++dt) != '$' )
|
|
|
|
str1[size_t(dt-str1.c_str()-1)] = '_';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
n->SetContent(str1);
|
|
|
|
}
|
|
|
|
n = n->GetNext();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void wxXmlRcEditDocument::UpgradeNode(wxXmlNode *node)
|
|
|
|
{
|
|
|
|
if (node)
|
|
|
|
{
|
|
|
|
UpgradeNodeValue(node);
|
|
|
|
UpgradeNode(node->GetNext());
|
|
|
|
UpgradeNode(node->GetChildren());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void wxXmlRcEditDocument::Upgrade()
|
|
|
|
{
|
|
|
|
int v1,v2,v3,v4;
|
|
|
|
long version;
|
|
|
|
wxXmlNode *node = GetRoot();
|
|
|
|
wxString verstr = wxT("0.0.0.0");
|
|
|
|
node->GetPropVal(wxT("version"),verstr);
|
|
|
|
if (wxSscanf(verstr.c_str(), wxT("%i.%i.%i.%i"),
|
|
|
|
&v1, &v2, &v3, &v4) == 4)
|
|
|
|
version = v1*256*256*256+v2*256*256+v3*256+v4;
|
|
|
|
else
|
|
|
|
version = 0;
|
|
|
|
if (!version)
|
|
|
|
{
|
|
|
|
UpgradeNode(node);
|
|
|
|
}
|
|
|
|
node->DeleteProperty(wxT("version"));
|
2002-12-21 00:20:35 +00:00
|
|
|
node->AddProperty(wxT("version"), WX_XMLRES_CURRENT_VERSION_STRING);
|
2001-05-29 18:21:09 +00:00
|
|
|
}
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
class EditorTreeCtrl : public wxTreeCtrl
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
EditorTreeCtrl(wxWindow *parent, int id, EditorFrame *frame)
|
|
|
|
: wxTreeCtrl(parent, id), m_EdFrame(frame) {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
EditorFrame *m_EdFrame;
|
|
|
|
|
|
|
|
void OnRightClick(wxMouseEvent &event)
|
|
|
|
{
|
|
|
|
wxTreeItemId item =
|
|
|
|
m_EdFrame->m_TreeCtrl->HitTest(event.GetPosition());
|
|
|
|
if (item.IsOk())
|
|
|
|
{
|
|
|
|
m_EdFrame->m_TreeCtrl->SelectItem(item);
|
|
|
|
m_EdFrame->OnRightClickTree(event.GetPosition());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DECLARE_EVENT_TABLE()
|
|
|
|
};
|
|
|
|
|
|
|
|
BEGIN_EVENT_TABLE(EditorTreeCtrl, wxTreeCtrl)
|
|
|
|
EVT_RIGHT_DOWN(EditorTreeCtrl::OnRightClick)
|
|
|
|
END_EVENT_TABLE()
|
|
|
|
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
ID_PREVIEW = wxID_HIGHEST + 100,
|
|
|
|
ID_NEW,
|
|
|
|
ID_OPEN,
|
|
|
|
ID_CLOSE,
|
|
|
|
ID_SAVE,
|
|
|
|
ID_SAVEAS,
|
|
|
|
ID_DELETE_NODE,
|
|
|
|
ID_EXIT,
|
|
|
|
ID_TREE,
|
2000-08-06 16:51:32 +00:00
|
|
|
|
|
|
|
ID_CUT,
|
|
|
|
ID_PASTE_SYBLING,
|
|
|
|
ID_PASTE_CHILD,
|
|
|
|
ID_COPY,
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
ID_NEWDIALOG,
|
2002-12-21 00:20:35 +00:00
|
|
|
ID_NEWFRAME,
|
2000-07-25 18:47:21 +00:00
|
|
|
ID_NEWPANEL,
|
|
|
|
ID_NEWMENU,
|
|
|
|
ID_NEWMENUBAR,
|
|
|
|
ID_NEWTOOLBAR,
|
2002-01-27 23:40:57 +00:00
|
|
|
ID_NEWNODE = wxID_HIGHEST + 10000, // safely out of XRCID range :)
|
2000-11-04 23:34:33 +00:00
|
|
|
ID_NEWSYBNODE = ID_NEWNODE + 20000
|
2000-07-25 18:47:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_EVENT_TABLE(EditorFrame, wxFrame)
|
|
|
|
EVT_TREE_SEL_CHANGED(ID_TREE, EditorFrame::OnTreeSel)
|
|
|
|
EVT_TOOL_RANGE(ID_PREVIEW, ID_EXIT, EditorFrame::OnToolbar)
|
|
|
|
EVT_MENU_RANGE(ID_NEWDIALOG, ID_NEWSYBNODE + 1000, EditorFrame::OnNewNode)
|
2000-08-06 16:51:32 +00:00
|
|
|
EVT_MENU_RANGE(ID_CUT, ID_COPY, EditorFrame::OnClipboardAction)
|
2000-11-24 17:19:48 +00:00
|
|
|
EVT_CLOSE(EditorFrame::OnCloseWindow)
|
2000-07-25 18:47:21 +00:00
|
|
|
END_EVENT_TABLE()
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-07-27 09:35:42 +00:00
|
|
|
#if defined(__UNIX__)
|
2000-07-25 18:47:21 +00:00
|
|
|
#include "bitmaps/preview.xpm"
|
|
|
|
#include "bitmaps/close.xpm"
|
|
|
|
#include "bitmaps/save.xpm"
|
|
|
|
#include "bitmaps/open.xpm"
|
|
|
|
|
|
|
|
#include "bitmaps/control.xpm"
|
|
|
|
#include "bitmaps/vsizer.xpm"
|
|
|
|
#include "bitmaps/hsizer.xpm"
|
|
|
|
#include "bitmaps/panel.xpm"
|
2000-07-26 21:49:18 +00:00
|
|
|
#include "bitmaps/gsizer.xpm"
|
|
|
|
#include "bitmaps/resicon.xpm"
|
2000-07-25 18:47:21 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
EditorFrame *EditorFrame::ms_Instance = NULL;
|
|
|
|
|
|
|
|
EditorFrame::EditorFrame(wxFrame *parent, const wxString& filename)
|
|
|
|
: wxFrame(parent, -1, filename + _("- wxWindows resources editor"))
|
|
|
|
{
|
|
|
|
ms_Instance = this;
|
2000-08-06 16:51:32 +00:00
|
|
|
|
|
|
|
m_Clipboard = NULL;
|
2000-11-24 17:19:48 +00:00
|
|
|
m_Modified = FALSE;
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
wxConfigBase *cfg = wxConfigBase::Get();
|
|
|
|
|
2003-09-18 13:47:05 +00:00
|
|
|
SetSize(wxRect(wxPoint(cfg->Read(_T("editor_x"), -1), cfg->Read(_T("editor_y"), -1)),
|
|
|
|
wxSize(cfg->Read(_T("editor_w"), 400), cfg->Read(_T("editor_h"), 400))));
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
m_SelectedNode = NULL;
|
|
|
|
m_Resource = NULL;
|
|
|
|
m_FileName = wxEmptyString;
|
|
|
|
|
|
|
|
wxMenu *menuFile = new wxMenu;
|
2003-09-18 13:47:05 +00:00
|
|
|
menuFile->Append(ID_NEW, _T("&New"));
|
|
|
|
menuFile->Append(ID_OPEN, _T("&Open\tCtrl-O"));
|
|
|
|
menuFile->Append(ID_SAVE, _T("&Save\tCtrl-S"));
|
|
|
|
menuFile->Append(ID_SAVEAS, _T("Save &as..."));
|
2000-07-25 18:47:21 +00:00
|
|
|
menuFile->AppendSeparator();
|
2003-09-18 13:47:05 +00:00
|
|
|
menuFile->Append(ID_EXIT, _T("E&xit\tAlt-X"));
|
2000-08-06 16:51:32 +00:00
|
|
|
|
|
|
|
wxMenu *menuEdit = new wxMenu;
|
2003-09-18 13:47:05 +00:00
|
|
|
menuEdit->Append(ID_CUT, _T("Cut\tCtrl-X"));
|
|
|
|
menuEdit->Append(ID_COPY, _T("Copy\tCtrl-C"));
|
|
|
|
menuEdit->Append(ID_PASTE_SYBLING, _T("Paste as sybling\tCtrl-V"));
|
|
|
|
menuEdit->Append(ID_PASTE_CHILD, _T("Paste as child"));
|
2000-08-06 16:51:32 +00:00
|
|
|
menuEdit->AppendSeparator();
|
2003-09-18 13:47:05 +00:00
|
|
|
menuEdit->Append(ID_DELETE_NODE, _T("Delete"));
|
2000-08-06 16:51:32 +00:00
|
|
|
|
|
|
|
menuEdit->Enable(ID_PASTE_SYBLING, FALSE);
|
|
|
|
menuEdit->Enable(ID_PASTE_CHILD, FALSE);
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
wxMenuBar *menuBar = new wxMenuBar();
|
2003-09-18 13:47:05 +00:00
|
|
|
menuBar->Append(menuFile, _T("&File"));
|
|
|
|
menuBar->Append(menuEdit, _T("&Edit"));
|
2000-07-25 18:47:21 +00:00
|
|
|
SetMenuBar(menuBar);
|
2000-08-06 16:51:32 +00:00
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
// Create toolbar:
|
|
|
|
wxToolBar *toolBar = CreateToolBar(wxNO_BORDER | wxTB_HORIZONTAL | wxTB_FLAT);
|
|
|
|
toolBar->SetMargins(2, 2);
|
2000-07-26 21:49:18 +00:00
|
|
|
toolBar->SetToolBitmapSize(wxSize(24, 24));
|
2000-07-25 18:47:21 +00:00
|
|
|
toolBar -> AddTool(ID_EXIT, wxBITMAP(close), wxNullBitmap,
|
|
|
|
FALSE, -1, -1, (wxObject *) NULL,
|
|
|
|
_("Quit the editor"));
|
|
|
|
toolBar -> AddTool(ID_OPEN, wxBITMAP(open), wxNullBitmap,
|
|
|
|
FALSE, -1, -1, (wxObject *) NULL,
|
|
|
|
_("Open XML resource file"));
|
|
|
|
toolBar -> AddTool(ID_SAVE, wxBITMAP(save), wxNullBitmap,
|
|
|
|
FALSE, -1, -1, (wxObject *) NULL,
|
|
|
|
_("Save XML file"));
|
|
|
|
toolBar -> AddTool(ID_PREVIEW, wxBITMAP(preview), wxNullBitmap,
|
|
|
|
FALSE, -1, -1, (wxObject *) NULL,
|
|
|
|
_("Preview"));
|
|
|
|
toolBar -> Realize();
|
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
wxSizer *sizer = new wxBoxSizer(wxHORIZONTAL);
|
2000-07-25 18:47:21 +00:00
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
// Create tree control:
|
|
|
|
m_TreeCtrl = new EditorTreeCtrl(this, ID_TREE, this);
|
|
|
|
wxImageList *imgList = new wxImageList(16, 16);
|
|
|
|
imgList->Add(wxICON(control));
|
|
|
|
imgList->Add(wxICON(panel));
|
|
|
|
imgList->Add(wxICON(vsizer));
|
|
|
|
imgList->Add(wxICON(hsizer));
|
|
|
|
imgList->Add(wxICON(gsizer));
|
|
|
|
imgList->Add(wxICON(resicon));
|
|
|
|
m_TreeCtrl->AssignImageList(imgList);
|
|
|
|
sizer->Add(m_TreeCtrl, 1, wxEXPAND);
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
SetAutoLayout(TRUE);
|
|
|
|
SetSizer(sizer);
|
|
|
|
|
|
|
|
// Load file:
|
|
|
|
if (!filename)
|
|
|
|
NewFile();
|
|
|
|
else
|
|
|
|
LoadFile(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
EditorFrame::~EditorFrame()
|
|
|
|
{
|
2000-10-07 21:56:37 +00:00
|
|
|
PreviewFrame::Get()->Close();
|
2000-11-04 23:34:33 +00:00
|
|
|
PropertiesFrame::Get()->Close();
|
2000-10-07 21:56:37 +00:00
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
wxConfigBase *cfg = wxConfigBase::Get();
|
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
cfg->Write(_T("editor_x"), (long)GetPosition().x);
|
|
|
|
cfg->Write(_T("editor_y"), (long)GetPosition().y);
|
|
|
|
cfg->Write(_T("editor_w"), (long)GetSize().x);
|
|
|
|
cfg->Write(_T("editor_h"), (long)GetSize().y);
|
|
|
|
|
2000-08-06 16:51:32 +00:00
|
|
|
delete m_Clipboard;
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::LoadFile(const wxString& filename)
|
|
|
|
{
|
2000-11-24 17:19:48 +00:00
|
|
|
if (!AskToSave()) return;
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
delete m_Resource;
|
|
|
|
|
2001-05-29 18:21:09 +00:00
|
|
|
// create new resource in order to handle version differences properly
|
|
|
|
PreviewFrame::Get()->ResetResource();
|
|
|
|
|
2003-09-18 13:47:05 +00:00
|
|
|
m_FileName = wxEmptyString;
|
2001-05-29 18:21:09 +00:00
|
|
|
m_Resource = new wxXmlRcEditDocument;
|
2000-11-24 17:19:48 +00:00
|
|
|
m_Modified = FALSE;
|
2000-07-25 18:47:21 +00:00
|
|
|
|
2002-02-02 23:17:38 +00:00
|
|
|
if (!m_Resource->Load(filename, wxLocale::GetSystemEncodingName()))
|
2000-07-25 18:47:21 +00:00
|
|
|
{
|
|
|
|
delete m_Resource;
|
|
|
|
m_Resource = NULL;
|
|
|
|
NewFile();
|
2003-09-18 13:47:05 +00:00
|
|
|
wxLogError(_T("Error parsing ") + filename);
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_FileName = filename;
|
2001-05-29 18:21:09 +00:00
|
|
|
|
|
|
|
// Upgrades old versions
|
|
|
|
m_Resource->Upgrade();
|
2000-07-25 18:47:21 +00:00
|
|
|
RefreshTree();
|
|
|
|
}
|
2000-11-24 17:19:48 +00:00
|
|
|
RefreshTitle();
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::SaveFile(const wxString& filename)
|
|
|
|
{
|
|
|
|
m_FileName = filename;
|
2001-05-06 22:21:43 +00:00
|
|
|
|
|
|
|
// save it:
|
2001-05-02 23:01:20 +00:00
|
|
|
if (!m_Resource->Save(filename))
|
2000-11-24 17:19:48 +00:00
|
|
|
wxLogError(_("Error saving ") + filename);
|
|
|
|
else
|
|
|
|
m_Modified = FALSE;
|
|
|
|
|
|
|
|
RefreshTitle();
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::NewFile()
|
|
|
|
{
|
2000-11-24 17:19:48 +00:00
|
|
|
if (!AskToSave()) return;
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
delete m_Resource;
|
|
|
|
|
2003-09-18 13:47:05 +00:00
|
|
|
m_FileName = wxEmptyString;
|
2001-05-29 18:21:09 +00:00
|
|
|
m_Resource = new wxXmlRcEditDocument;
|
2000-11-24 17:19:48 +00:00
|
|
|
m_Resource->SetRoot(new wxXmlNode(wxXML_ELEMENT_NODE, _("resource")));
|
2002-12-21 00:20:35 +00:00
|
|
|
|
2003-09-18 13:47:05 +00:00
|
|
|
m_Resource->SetFileEncoding(_T("utf-8"));
|
2002-12-21 13:35:13 +00:00
|
|
|
#if !wxUSE_UNICODE
|
|
|
|
m_Resource->SetEncoding(wxLocale::GetSystemEncodingName());
|
|
|
|
#endif
|
2000-07-25 18:47:21 +00:00
|
|
|
|
2002-12-21 00:20:35 +00:00
|
|
|
m_Resource->GetRoot()->AddProperty(_T("version"),
|
|
|
|
WX_XMLRES_CURRENT_VERSION_STRING);
|
|
|
|
|
2000-11-24 17:19:48 +00:00
|
|
|
m_Modified = FALSE;
|
2000-07-25 18:47:21 +00:00
|
|
|
RefreshTree();
|
2000-11-24 17:19:48 +00:00
|
|
|
RefreshTitle();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::RefreshTitle()
|
|
|
|
{
|
|
|
|
wxString s;
|
|
|
|
if (m_Modified) s << _T("* ");
|
|
|
|
s << _("wxrcedit");
|
2003-09-18 13:47:05 +00:00
|
|
|
if (m_FileName != wxEmptyString)
|
2000-11-24 17:19:48 +00:00
|
|
|
s << _T(" - ") << wxFileNameFromPath(m_FileName);
|
|
|
|
SetTitle(s);
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::RefreshTree()
|
|
|
|
{
|
|
|
|
wxXmlNode *sel = m_SelectedNode;
|
|
|
|
|
|
|
|
m_TreeCtrl->DeleteAllItems();
|
2002-12-21 00:20:35 +00:00
|
|
|
|
2003-09-18 13:47:05 +00:00
|
|
|
wxTreeItemId root = m_TreeCtrl->AddRoot(_T("Resource: ") + wxFileNameFromPath(m_FileName), 5, 5);
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
wxXmlNode *n = m_Resource->GetRoot()->GetChildren();
|
|
|
|
while (n)
|
|
|
|
{
|
|
|
|
if (n->GetType() == wxXML_ELEMENT_NODE)
|
|
|
|
CreateTreeNode(m_TreeCtrl, root, n);
|
|
|
|
n = n->GetNext();
|
|
|
|
}
|
|
|
|
|
|
|
|
m_TreeCtrl->Expand(root);
|
|
|
|
SelectNode(sel);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-11-25 23:14:27 +00:00
|
|
|
|
|
|
|
static void RecursivelyExpand(wxTreeCtrl *t, wxTreeItemId item)
|
|
|
|
{
|
|
|
|
t->Expand(item);
|
2003-09-10 20:53:01 +00:00
|
|
|
void* cookie;
|
2000-11-25 23:14:27 +00:00
|
|
|
wxTreeItemId id = t->GetFirstChild(item, cookie);
|
|
|
|
while (id.IsOk())
|
|
|
|
{
|
|
|
|
RecursivelyExpand(t, id);
|
|
|
|
id = t->GetNextChild(item, cookie);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
bool EditorFrame::SelectNode(wxXmlNode *node, wxTreeItemId *root)
|
|
|
|
{
|
|
|
|
if (root == NULL)
|
|
|
|
{
|
|
|
|
wxTreeItemId rootitem = m_TreeCtrl->GetRootItem();
|
|
|
|
return SelectNode(node, &rootitem);
|
|
|
|
}
|
|
|
|
|
|
|
|
wxTreeItemId item;
|
|
|
|
XmlTreeData *dt;
|
|
|
|
wxXmlNode *nd;
|
2003-09-10 20:53:01 +00:00
|
|
|
void* cookie;
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
item = m_TreeCtrl->GetFirstChild(*root, cookie);
|
|
|
|
while (item.IsOk())
|
|
|
|
{
|
|
|
|
dt = (XmlTreeData*)(m_TreeCtrl->GetItemData(item));
|
|
|
|
nd = (dt) ? dt->Node : NULL;
|
|
|
|
if (nd == node)
|
|
|
|
{
|
2000-11-25 23:14:27 +00:00
|
|
|
RecursivelyExpand(m_TreeCtrl, *root);
|
2000-07-25 18:47:21 +00:00
|
|
|
m_TreeCtrl->SelectItem(item);
|
|
|
|
m_TreeCtrl->EnsureVisible(item);
|
2000-11-25 23:14:27 +00:00
|
|
|
return TRUE;
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
if (m_TreeCtrl->ItemHasChildren(item) && SelectNode(node, &item))
|
2000-11-25 23:14:27 +00:00
|
|
|
return TRUE;
|
2000-07-25 18:47:21 +00:00
|
|
|
item = m_TreeCtrl->GetNextChild(*root, cookie);
|
|
|
|
}
|
2000-11-25 23:14:27 +00:00
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
wxTreeItemId EditorFrame::CreateTreeNode(wxTreeCtrl *treectrl, wxTreeItemId parent, wxXmlNode *node)
|
|
|
|
{
|
|
|
|
if (!node)
|
|
|
|
{
|
|
|
|
wxTreeItemId invalid;
|
|
|
|
return invalid;
|
|
|
|
}
|
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
return NodeHandler::Find(node)->CreateTreeNode(treectrl, parent, node);
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::NotifyChanged(int change_type)
|
|
|
|
{
|
|
|
|
if (change_type & CHANGED_TREE)
|
|
|
|
RefreshTree();
|
|
|
|
|
|
|
|
if (change_type & CHANGED_TREE_SELECTED)
|
|
|
|
{
|
|
|
|
wxTreeItemId sel = m_TreeCtrl->GetSelection();
|
|
|
|
m_TreeCtrl->SetItemText(sel,
|
2000-11-04 23:34:33 +00:00
|
|
|
NodeHandler::Find(m_SelectedNode)->GetTreeString(m_SelectedNode));
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (change_type & CHANGED_TREE_SELECTED_ICON)
|
|
|
|
{
|
|
|
|
wxTreeItemId sel = m_TreeCtrl->GetSelection();
|
2000-11-04 23:34:33 +00:00
|
|
|
int icon = NodeHandler::Find(m_SelectedNode)->GetTreeIcon(m_SelectedNode);
|
2000-07-25 18:47:21 +00:00
|
|
|
m_TreeCtrl->SetItemImage(sel, icon);
|
|
|
|
}
|
2000-11-24 17:19:48 +00:00
|
|
|
|
|
|
|
if (!m_Modified)
|
|
|
|
{
|
|
|
|
m_Modified = TRUE;
|
|
|
|
RefreshTitle();
|
|
|
|
}
|
|
|
|
|
|
|
|
PreviewFrame::Get()->MakeDirty();
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
void EditorFrame::OnTreeSel(wxTreeEvent& event)
|
2000-07-25 18:47:21 +00:00
|
|
|
{
|
2000-11-04 23:34:33 +00:00
|
|
|
XmlTreeData *dt = (XmlTreeData*)(m_TreeCtrl->GetItemData(event.GetItem()));
|
|
|
|
wxXmlNode *node = (dt) ? dt->Node : NULL;
|
|
|
|
|
|
|
|
m_SelectedNode = node;
|
|
|
|
if (node)
|
|
|
|
PropertiesFrame::Get()->ShowProps(node);
|
2000-07-25 18:47:21 +00:00
|
|
|
|
2002-12-29 07:48:21 +00:00
|
|
|
if (m_TreeCtrl->GetItemParent(event.GetItem()) == m_TreeCtrl->GetRootItem())
|
2000-07-25 18:47:21 +00:00
|
|
|
{
|
2000-11-04 23:34:33 +00:00
|
|
|
wxTreeItemId it = event.GetOldItem();
|
2000-07-25 18:47:21 +00:00
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
if (it.IsOk() && m_TreeCtrl->GetRootItem() != it)
|
2000-07-25 18:47:21 +00:00
|
|
|
{
|
2002-12-29 07:48:21 +00:00
|
|
|
while (m_TreeCtrl->GetItemParent(it) != m_TreeCtrl->GetRootItem())
|
|
|
|
it = m_TreeCtrl->GetItemParent(it);
|
2000-11-04 23:34:33 +00:00
|
|
|
m_TreeCtrl->Collapse(it);
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
2000-11-04 23:34:33 +00:00
|
|
|
RecursivelyExpand(m_TreeCtrl, event.GetItem());
|
2000-07-25 18:47:21 +00:00
|
|
|
|
2002-12-04 14:11:26 +00:00
|
|
|
PreviewFrame::Get()->Preview(node,m_Resource);
|
2000-11-04 23:34:33 +00:00
|
|
|
}
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::OnToolbar(wxCommandEvent& event)
|
|
|
|
{
|
|
|
|
switch (event.GetId())
|
|
|
|
{
|
|
|
|
case ID_PREVIEW :
|
|
|
|
{
|
|
|
|
XmlTreeData* dt = (XmlTreeData*)m_TreeCtrl->GetItemData(m_TreeCtrl->GetSelection());;
|
|
|
|
if (dt != NULL && dt->Node != NULL)
|
2002-12-04 14:11:26 +00:00
|
|
|
PreviewFrame::Get()->Preview(dt->Node, m_Resource);
|
2000-07-25 18:47:21 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ID_EXIT :
|
|
|
|
Close(TRUE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_NEW :
|
|
|
|
NewFile();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_OPEN :
|
|
|
|
{
|
2000-11-24 17:19:48 +00:00
|
|
|
wxString cwd = wxGetCwd(); // workaround for 2.2
|
2000-11-04 23:34:33 +00:00
|
|
|
wxString name = wxFileSelector(_("Open XML resource"), _T(""), _T(""), _T(""), _("XML resources (*.xrc)|*.xrc"), wxOPEN | wxFILE_MUST_EXIST);
|
2000-11-24 17:19:48 +00:00
|
|
|
wxSetWorkingDirectory(cwd);
|
2000-07-25 18:47:21 +00:00
|
|
|
if (!name.IsEmpty())
|
|
|
|
LoadFile(name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ID_SAVE :
|
2003-09-18 13:47:05 +00:00
|
|
|
if (m_FileName != wxEmptyString) { SaveFile(m_FileName); break;}
|
2000-07-25 18:47:21 +00:00
|
|
|
// else go to SAVEAS
|
|
|
|
|
|
|
|
case ID_SAVEAS :
|
|
|
|
{
|
2000-11-24 17:19:48 +00:00
|
|
|
wxString cwd = wxGetCwd(); // workaround for 2.2
|
2000-11-04 23:34:33 +00:00
|
|
|
wxString name = wxFileSelector(_("Save as"), _T(""), m_FileName, _T(""), _("XML resources (*.xrc)|*.xrc"), wxSAVE | wxOVERWRITE_PROMPT);
|
2000-11-24 17:19:48 +00:00
|
|
|
wxSetWorkingDirectory(cwd);
|
2000-07-25 18:47:21 +00:00
|
|
|
if (!name.IsEmpty())
|
|
|
|
SaveFile((m_FileName = name));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ID_DELETE_NODE :
|
|
|
|
{
|
2000-08-06 16:51:32 +00:00
|
|
|
DeleteSelectedNode();
|
2000-07-25 18:47:21 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-08-06 16:51:32 +00:00
|
|
|
void EditorFrame::DeleteSelectedNode()
|
|
|
|
{
|
|
|
|
XmlTreeData *dt = (XmlTreeData*)
|
2002-12-29 07:48:21 +00:00
|
|
|
(m_TreeCtrl->GetItemData(m_TreeCtrl->GetItemParent(m_TreeCtrl->GetSelection())));
|
2000-08-06 16:51:32 +00:00
|
|
|
wxXmlNode *n = (dt) ? dt->Node : NULL;
|
|
|
|
|
|
|
|
m_SelectedNode->GetParent()->RemoveChild(m_SelectedNode);
|
|
|
|
NotifyChanged(CHANGED_TREE);
|
|
|
|
SelectNode(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
void EditorFrame::OnNewNode(wxCommandEvent& event)
|
|
|
|
{
|
|
|
|
if (event.GetId() >= ID_NEWSYBNODE)
|
|
|
|
{
|
|
|
|
XmlTreeData *pardt =
|
|
|
|
(XmlTreeData*)(m_TreeCtrl->GetItemData(
|
2002-12-29 07:48:21 +00:00
|
|
|
m_TreeCtrl->GetItemParent(m_TreeCtrl->GetSelection())));
|
2000-07-25 18:47:21 +00:00
|
|
|
|
|
|
|
if (pardt && pardt->Node && pardt->Node != m_Resource->GetRoot())
|
|
|
|
{
|
|
|
|
wxXmlNode *nd = pardt->Node;
|
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
wxXmlNode *realnode = NodeHandler::Find(nd)->GetRealNode(nd);
|
|
|
|
NodeHandler *hnd = NodeHandler::Find(realnode);
|
2000-07-25 18:47:21 +00:00
|
|
|
wxString name = hnd->GetChildTypes()[event.GetId()-ID_NEWSYBNODE];
|
|
|
|
|
2000-10-07 21:56:37 +00:00
|
|
|
wxXmlNode *node = new wxXmlNode(wxXML_ELEMENT_NODE, _T("object"));
|
|
|
|
node->AddProperty(_T("class"), name);
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
hnd->InsertNode(realnode, node, m_SelectedNode);
|
|
|
|
wxTreeItemId root = m_TreeCtrl->GetSelection();
|
|
|
|
SelectNode(node, &root);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (event.GetId() >= ID_NEWNODE)
|
|
|
|
{
|
2000-11-04 23:34:33 +00:00
|
|
|
wxXmlNode *realnode = NodeHandler::Find(m_SelectedNode)->GetRealNode(m_SelectedNode);
|
|
|
|
NodeHandler *hnd = NodeHandler::Find(realnode);
|
2000-07-25 18:47:21 +00:00
|
|
|
wxString name = hnd->GetChildTypes()[event.GetId()-ID_NEWNODE];
|
|
|
|
|
2000-10-07 21:56:37 +00:00
|
|
|
wxXmlNode *node = new wxXmlNode(wxXML_ELEMENT_NODE, _T("object"));
|
|
|
|
node->AddProperty(_T("class"), name);
|
|
|
|
|
2000-07-25 18:47:21 +00:00
|
|
|
hnd->InsertNode(realnode, node);
|
|
|
|
wxTreeItemId root = m_TreeCtrl->GetSelection();
|
|
|
|
SelectNode(node, &root);
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
wxString name;
|
|
|
|
switch (event.GetId())
|
|
|
|
{
|
2000-10-07 21:56:37 +00:00
|
|
|
case ID_NEWDIALOG : name = _T("wxDialog"); break;
|
2002-12-21 00:20:35 +00:00
|
|
|
case ID_NEWFRAME : name = _T("wxFrame"); break;
|
2000-10-07 21:56:37 +00:00
|
|
|
case ID_NEWPANEL : name = _T("wxPanel"); break;
|
|
|
|
case ID_NEWMENU : name = _T("wxMenu"); break;
|
|
|
|
case ID_NEWMENUBAR : name = _T("wxMenuBar"); break;
|
|
|
|
case ID_NEWTOOLBAR : name = _T("wxToolBar"); break;
|
2000-07-25 18:47:21 +00:00
|
|
|
default : return; // never occurs
|
|
|
|
}
|
|
|
|
|
2000-10-07 21:56:37 +00:00
|
|
|
wxXmlNode *node = new wxXmlNode(wxXML_ELEMENT_NODE, _T("object"));
|
|
|
|
node->AddProperty(_T("class"), name);
|
2000-07-25 18:47:21 +00:00
|
|
|
m_Resource->GetRoot()->AddChild(node);
|
|
|
|
NotifyChanged(CHANGED_TREE);
|
|
|
|
SelectNode(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::OnRightClickTree(wxPoint pos)
|
|
|
|
{
|
|
|
|
wxMenu *popup = new wxMenu;
|
|
|
|
|
|
|
|
if (m_SelectedNode == NULL || m_SelectedNode == m_Resource->GetRoot())
|
|
|
|
{
|
2000-10-07 21:56:37 +00:00
|
|
|
popup->Append(ID_NEWDIALOG, _("New wxDialog"));
|
2002-12-21 00:20:35 +00:00
|
|
|
popup->Append(ID_NEWFRAME, _("New wxFrame"));
|
2000-10-07 21:56:37 +00:00
|
|
|
popup->Append(ID_NEWPANEL, _("New wxPanel"));
|
|
|
|
popup->Append(ID_NEWMENU, _("New wxMenu"));
|
|
|
|
popup->Append(ID_NEWMENUBAR, _("New wxMenuBar"));
|
|
|
|
popup->Append(ID_NEWTOOLBAR, _("New wxToolBar"));
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
2000-08-06 16:51:32 +00:00
|
|
|
bool has_children;
|
2000-07-25 18:47:21 +00:00
|
|
|
{
|
|
|
|
wxArrayString& arr =
|
2000-11-04 23:34:33 +00:00
|
|
|
NodeHandler::Find(NodeHandler::Find(m_SelectedNode)->GetRealNode(m_SelectedNode))->
|
2000-07-25 18:47:21 +00:00
|
|
|
GetChildTypes();
|
|
|
|
|
2000-08-06 16:51:32 +00:00
|
|
|
has_children = !arr.IsEmpty();
|
2000-07-25 18:47:21 +00:00
|
|
|
if (!arr.IsEmpty())
|
|
|
|
{
|
|
|
|
wxMenu *news = new wxMenu;
|
2000-10-07 21:56:37 +00:00
|
|
|
wxMenu *news2 = news;
|
2000-07-25 18:47:21 +00:00
|
|
|
for (size_t i = 0; i < arr.GetCount(); i++)
|
|
|
|
{
|
2000-10-07 21:56:37 +00:00
|
|
|
news2->Append(i + ID_NEWNODE, arr[i]);
|
|
|
|
#ifdef __WXGTK__ // doesn't support Break
|
|
|
|
if (i % 20 == 19)
|
|
|
|
{
|
|
|
|
wxMenu *m = new wxMenu;
|
|
|
|
news2->Append(ID_NEWNODE+arr.GetCount(), _("More..."), m);
|
|
|
|
news2 = m;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (i % 16 == 15) news2->Break();
|
|
|
|
#endif
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
popup->Append(ID_NEWNODE-1, _("New child"), news);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
XmlTreeData *pardt =
|
|
|
|
(XmlTreeData*)(m_TreeCtrl->GetItemData(
|
2002-12-29 07:48:21 +00:00
|
|
|
m_TreeCtrl->GetItemParent(m_TreeCtrl->GetSelection())));
|
2000-07-25 18:47:21 +00:00
|
|
|
if (pardt && pardt->Node && pardt->Node != m_Resource->GetRoot())
|
|
|
|
{
|
|
|
|
wxXmlNode *nd = pardt->Node;
|
|
|
|
wxArrayString& arr =
|
2000-11-04 23:34:33 +00:00
|
|
|
NodeHandler::Find(NodeHandler::Find(nd)->GetRealNode(nd))->
|
2000-07-25 18:47:21 +00:00
|
|
|
GetChildTypes();
|
|
|
|
|
|
|
|
if (!arr.IsEmpty())
|
|
|
|
{
|
|
|
|
wxMenu *news = new wxMenu;
|
2000-10-07 21:56:37 +00:00
|
|
|
wxMenu *news2 = news;
|
2000-07-25 18:47:21 +00:00
|
|
|
for (size_t i = 0; i < arr.GetCount(); i++)
|
|
|
|
{
|
2000-10-07 21:56:37 +00:00
|
|
|
news2->Append(i + ID_NEWSYBNODE, arr[i]);
|
|
|
|
#ifdef __WXGTK__ // doesn't support Break
|
|
|
|
if (i % 20 == 19)
|
|
|
|
{
|
|
|
|
wxMenu *m = new wxMenu;
|
|
|
|
news2->Append(ID_NEWSYBNODE+arr.GetCount(), _("More..."), m);
|
|
|
|
news2 = m;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (i % 16 == 15) news2->Break();
|
|
|
|
#endif
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
popup->Append(ID_NEWSYBNODE-1, _("New sybling"), news);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-08-06 16:51:32 +00:00
|
|
|
popup->AppendSeparator();
|
2000-10-07 21:56:37 +00:00
|
|
|
popup->Append(ID_CUT, _("Cut"));
|
|
|
|
popup->Append(ID_COPY, _("Copy"));
|
|
|
|
popup->Append(ID_PASTE_SYBLING, _("Paste as sybling"));
|
|
|
|
popup->Append(ID_PASTE_CHILD, _("Paste as child"));
|
2000-07-25 18:47:21 +00:00
|
|
|
popup->AppendSeparator();
|
|
|
|
popup->Append(ID_DELETE_NODE, _("Delete"));
|
2000-08-06 16:51:32 +00:00
|
|
|
popup->Enable(ID_PASTE_SYBLING, m_Clipboard != NULL);
|
|
|
|
popup->Enable(ID_PASTE_CHILD, has_children && m_Clipboard != NULL);
|
2000-07-25 18:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m_TreeCtrl->PopupMenu(popup, pos);
|
|
|
|
delete popup;
|
|
|
|
}
|
2000-08-06 16:51:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::OnClipboardAction(wxCommandEvent& event)
|
|
|
|
{
|
|
|
|
switch (event.GetId())
|
|
|
|
{
|
|
|
|
case ID_COPY:
|
|
|
|
case ID_CUT:
|
|
|
|
delete m_Clipboard;
|
|
|
|
m_Clipboard = new wxXmlNode(*m_SelectedNode);
|
|
|
|
GetMenuBar()->Enable(ID_PASTE_SYBLING, TRUE);
|
|
|
|
GetMenuBar()->Enable(ID_PASTE_CHILD, TRUE);
|
|
|
|
if (event.GetId() == ID_CUT) DeleteSelectedNode();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_PASTE_SYBLING:
|
|
|
|
{
|
|
|
|
XmlTreeData *pardt =
|
|
|
|
(XmlTreeData*)(m_TreeCtrl->GetItemData(
|
2002-12-29 07:48:21 +00:00
|
|
|
m_TreeCtrl->GetItemParent(m_TreeCtrl->GetSelection())));
|
2000-08-06 16:51:32 +00:00
|
|
|
|
|
|
|
if (pardt && pardt->Node && pardt->Node != m_Resource->GetRoot())
|
|
|
|
{
|
|
|
|
wxXmlNode *nd = pardt->Node;
|
|
|
|
|
2000-11-04 23:34:33 +00:00
|
|
|
wxXmlNode *realnode = NodeHandler::Find(nd)->GetRealNode(nd);
|
|
|
|
NodeHandler *hnd = NodeHandler::Find(realnode);
|
2000-08-06 16:51:32 +00:00
|
|
|
wxXmlNode *node = new wxXmlNode(*m_Clipboard);
|
|
|
|
hnd->InsertNode(realnode, node, m_SelectedNode);
|
|
|
|
wxTreeItemId root = m_TreeCtrl->GetSelection();
|
|
|
|
SelectNode(node, &root);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_PASTE_CHILD:
|
2000-11-04 23:34:33 +00:00
|
|
|
wxXmlNode *realnode = NodeHandler::Find(m_SelectedNode)->GetRealNode(m_SelectedNode);
|
|
|
|
NodeHandler *hnd = NodeHandler::Find(realnode);
|
2000-08-06 16:51:32 +00:00
|
|
|
wxXmlNode *node = new wxXmlNode(*m_Clipboard);
|
|
|
|
hnd->InsertNode(realnode, node);
|
|
|
|
wxTreeItemId root = m_TreeCtrl->GetSelection();
|
|
|
|
SelectNode(node, &root);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-11-24 17:19:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool EditorFrame::AskToSave()
|
|
|
|
// asks the user to save current document (if modified)
|
|
|
|
// returns FALSE if user cancelled the action, TRUE of he choosed
|
|
|
|
// 'yes' or 'no'
|
|
|
|
{
|
|
|
|
if (!m_Modified) return TRUE;
|
|
|
|
|
|
|
|
int res = wxMessageBox(_("File modified. Do you want to save changes?"), _("Save changes"),
|
|
|
|
wxYES_NO | wxCANCEL | wxCENTRE | wxICON_QUESTION);
|
|
|
|
if (res == wxYES)
|
|
|
|
SaveFile(m_FileName);
|
|
|
|
return (res != wxCANCEL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EditorFrame::OnCloseWindow(wxCloseEvent&)
|
|
|
|
{
|
|
|
|
if (!AskToSave()) return;
|
|
|
|
Destroy();
|
|
|
|
}
|