2015-04-02 18:00:50 -04:00
|
|
|
#include "EditPane.h"
|
|
|
|
|
|
|
|
EditPane::EditPane(
|
|
|
|
wxWindow *parent, wxWindowID id, const wxPoint &pos,
|
|
|
|
const wxSize &size, long style
|
|
|
|
) : wxStyledTextCtrl (parent, id, pos, size, style)
|
2015-04-13 16:48:41 -04:00
|
|
|
{
|
2015-04-14 17:00:48 -04:00
|
|
|
config = new TyroConfig();
|
2015-04-16 17:03:27 -04:00
|
|
|
|
2015-04-16 11:23:08 -04:00
|
|
|
lexerMap["batch"] = wxSTC_LEX_BATCH;
|
|
|
|
lexerMap["caml"] = wxSTC_LEX_CAML;
|
|
|
|
lexerMap["cmake"] = wxSTC_LEX_CMAKE;
|
2015-04-20 16:35:51 -04:00
|
|
|
lexerMap["cobol"] = wxSTC_LEX_COBOL;
|
|
|
|
lexerMap["coffeescript"] = wxSTC_LEX_CPP;
|
2015-04-16 11:23:08 -04:00
|
|
|
lexerMap["cpp"] = wxSTC_LEX_CPP;
|
|
|
|
lexerMap["css"] = wxSTC_LEX_CSS;
|
|
|
|
lexerMap["js"] = wxSTC_LEX_CPP;
|
|
|
|
lexerMap["html"] = wxSTC_LEX_HTML;
|
|
|
|
lexerMap["makefile"] = wxSTC_LEX_MAKEFILE;
|
2015-04-20 16:35:51 -04:00
|
|
|
lexerMap["php"] = wxSTC_LEX_PHPSCRIPT;
|
2015-04-16 11:23:08 -04:00
|
|
|
lexerMap["perl"] = wxSTC_LEX_PERL;
|
|
|
|
lexerMap["python"] = wxSTC_LEX_PYTHON;
|
|
|
|
lexerMap["shell"] = wxSTC_LEX_BASH;
|
2015-04-16 17:03:27 -04:00
|
|
|
|
|
|
|
this->BindEvents();
|
2015-04-02 18:00:50 -04:00
|
|
|
}
|
|
|
|
|
2015-04-14 15:58:43 -04:00
|
|
|
EditPane::~EditPane()
|
|
|
|
{
|
|
|
|
delete config;
|
|
|
|
}
|
2015-04-10 15:11:15 -04:00
|
|
|
|
|
|
|
/**
|
2015-04-17 16:55:48 -04:00
|
|
|
* Handle the highlighting config for the
|
|
|
|
* selected file
|
|
|
|
*
|
2015-04-10 15:11:15 -04:00
|
|
|
* @param wxString filePath
|
2015-04-17 16:55:48 -04:00
|
|
|
* @return void
|
2015-04-10 15:11:15 -04:00
|
|
|
*/
|
2015-04-17 16:55:48 -04:00
|
|
|
void EditPane::Highlight(wxString filePath)
|
2015-04-10 15:11:15 -04:00
|
|
|
{
|
2015-04-17 16:55:48 -04:00
|
|
|
this->fileName.Assign(filePath);
|
2015-04-13 13:01:25 -04:00
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
wxLogDebug("Highlighting method");
|
|
|
|
|
|
|
|
// Get the configuration name for the selected language
|
|
|
|
string lang = this->GetLangByFile();
|
|
|
|
|
2015-04-14 15:58:43 -04:00
|
|
|
this->StyleClearAll();
|
2015-04-16 11:23:08 -04:00
|
|
|
|
2015-04-16 17:03:27 -04:00
|
|
|
// Font setup
|
2015-04-17 12:57:49 -04:00
|
|
|
#ifdef __WXMAC__
|
2015-04-16 17:03:27 -04:00
|
|
|
wxFont *defaultFont = wxFont::New(
|
|
|
|
14,
|
|
|
|
wxFONTFAMILY_MODERN,
|
2015-04-17 12:57:49 -04:00
|
|
|
wxFONTFLAG_ANTIALIASED
|
2015-04-16 17:03:27 -04:00
|
|
|
);
|
2015-04-17 12:57:49 -04:00
|
|
|
#else
|
|
|
|
wxFont *defaultFont = wxFont::New(
|
|
|
|
12,
|
|
|
|
wxFONTFAMILY_MODERN,
|
|
|
|
wxFONTFLAG_ANTIALIASED
|
|
|
|
);
|
|
|
|
#endif
|
2015-04-16 17:03:27 -04:00
|
|
|
|
2015-04-16 11:23:08 -04:00
|
|
|
if (lexerMap.count(lang) > 0)
|
|
|
|
{
|
|
|
|
this->SetLexer(lexerMap[lang]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this->SetLexer(wxSTC_LEX_NULL);
|
|
|
|
}
|
|
|
|
|
2015-04-17 12:48:00 -04:00
|
|
|
// Some basic properties to set
|
|
|
|
this->SetProperty("technology", "2");
|
|
|
|
this->SetProperty("error.inline", "0");
|
|
|
|
this->SetProperty("font.quality", "3"); // LCD Optimized
|
|
|
|
|
2015-04-16 11:23:08 -04:00
|
|
|
// Get the list of keywords for the current language
|
|
|
|
JsonValue keywords_array = config->GetLangKeywords(lang);
|
2015-04-14 08:09:11 -04:00
|
|
|
|
2015-04-14 15:58:43 -04:00
|
|
|
// Make sure every background is the same color!
|
|
|
|
for(int i = 0; i <= wxSTC_STYLE_MAX; i++)
|
|
|
|
{
|
|
|
|
this->StyleSetBackground(i, wxColor(253, 246, 227));
|
2015-04-16 17:03:27 -04:00
|
|
|
this->StyleSetFont(i, *defaultFont);
|
2015-04-14 15:58:43 -04:00
|
|
|
}
|
|
|
|
|
2015-04-13 13:01:25 -04:00
|
|
|
this->StyleSetForeground (wxSTC_STYLE_DEFAULT, wxColor(101, 123, 131));
|
2015-04-14 15:58:43 -04:00
|
|
|
this->StyleSetForeground(wxSTC_STYLE_INDENTGUIDE, wxColor(147, 161, 161));
|
2015-04-13 13:01:25 -04:00
|
|
|
|
2015-04-16 17:03:27 -04:00
|
|
|
this->SetMarginWidth (MARGIN_LINE_NUMBERS, TextWidth(wxSTC_STYLE_LINENUMBER, _T("_9999")));
|
2015-04-13 13:01:25 -04:00
|
|
|
this->StyleSetForeground (wxSTC_STYLE_LINENUMBER, wxColor(147, 161, 161));
|
|
|
|
this->StyleSetBackground (wxSTC_STYLE_LINENUMBER, wxColor(238, 232, 213));
|
|
|
|
this->SetMarginType (MARGIN_LINE_NUMBERS, wxSTC_MARGIN_NUMBER);
|
|
|
|
|
2015-04-14 15:58:43 -04:00
|
|
|
this->StyleSetForeground(wxSTC_C_DEFAULT, wxColor(101, 123, 131));
|
|
|
|
this->StyleSetForeground(wxSTC_C_OPERATOR, wxColor(101, 123, 131));
|
|
|
|
this->StyleSetForeground(wxSTC_C_HASHQUOTEDSTRING, wxColor(181, 137, 0));
|
|
|
|
this->StyleSetForeground(wxSTC_C_STRING, wxColor(108, 113, 196));
|
|
|
|
this->StyleSetForeground(wxSTC_C_PREPROCESSOR, wxColor(181, 137, 0));
|
|
|
|
this->StyleSetForeground(wxSTC_C_PREPROCESSORCOMMENT, wxColor(181, 137, 0));
|
|
|
|
this->StyleSetForeground(wxSTC_C_IDENTIFIER, wxColor(38, 139, 210));
|
|
|
|
this->StyleSetForeground(wxSTC_C_NUMBER, wxColor(211, 54, 130));
|
|
|
|
this->StyleSetForeground(wxSTC_C_CHARACTER, wxColor(108, 113, 196));
|
|
|
|
this->StyleSetForeground(wxSTC_C_WORD2, wxColor(203, 75, 22));
|
|
|
|
this->StyleSetForeground(wxSTC_C_WORD, wxColor(220, 50, 47));
|
|
|
|
this->StyleSetForeground(wxSTC_C_COMMENT, wxColor(147, 161, 161));
|
|
|
|
this->StyleSetForeground(wxSTC_C_COMMENTLINE, wxColor(147, 161, 161));
|
|
|
|
this->StyleSetForeground(wxSTC_C_COMMENTLINEDOC, wxColor(147, 161, 161));
|
|
|
|
this->StyleSetForeground(wxSTC_C_COMMENTDOC, wxColor(147, 161, 161));
|
|
|
|
this->StyleSetForeground(wxSTC_C_COMMENTDOCKEYWORD, wxColor(131, 148, 150));
|
|
|
|
this->StyleSetForeground(wxSTC_C_COMMENTDOCKEYWORDERROR, wxColor(101, 123, 131));
|
|
|
|
this->StyleSetBold(wxSTC_C_WORD, false);
|
2015-04-13 13:01:25 -04:00
|
|
|
this->StyleSetBold(wxSTC_C_WORD2, true);
|
2015-04-13 16:48:41 -04:00
|
|
|
this->StyleSetBold(wxSTC_C_COMMENTDOCKEYWORD, true);
|
2015-04-14 15:58:43 -04:00
|
|
|
this->StyleSetBold(wxSTC_C_OPERATOR, true);
|
|
|
|
|
2015-04-16 17:03:27 -04:00
|
|
|
// Set up Code folding
|
|
|
|
this->SetProperty(wxT("fold"), wxT("1"));
|
|
|
|
this->SetProperty(wxT("fold.comment"), wxT("1"));
|
|
|
|
this->SetProperty(wxT("fold.compact"), wxT("1"));
|
|
|
|
this->SetProperty(wxT("fold.html"), wxT("1"));
|
|
|
|
this->SetFoldFlags(wxSTC_FOLDFLAG_LINEBEFORE_CONTRACTED | wxSTC_FOLDFLAG_LINEAFTER_CONTRACTED);
|
|
|
|
this->SetMarginType(MARGIN_FOLD, wxSTC_MARGIN_SYMBOL);
|
|
|
|
this->SetMarginWidth(MARGIN_FOLD, 16);
|
|
|
|
this->SetMarginSensitive(MARGIN_FOLD, true);
|
|
|
|
this->SetMarginMask(MARGIN_FOLD, wxSTC_MASK_FOLDERS);
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDER, wxSTC_MARK_BOXPLUSCONNECTED, _T("WHITE"), _T("BLACK"));
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDEROPEN, wxSTC_MARK_BOXMINUSCONNECTED, _T("WHITE"), _T("BLACK"));
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDERSUB, wxSTC_MARK_VLINE, _T("BLACK"), _T("BLACK"));
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDEREND, wxSTC_MARK_CIRCLEPLUSCONNECTED, _T("WHITE"), _T("BLACK"));
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDEROPENMID, wxSTC_MARK_CIRCLEMINUSCONNECTED, _T("WHITE"), _T("BLACK"));
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDERMIDTAIL, wxSTC_MARK_TCORNER, _T("BLACK"), _T("BLACK"));
|
|
|
|
this->MarkerDefine (wxSTC_MARKNUM_FOLDERTAIL, wxSTC_MARK_LCORNER, _T("BLACK"), _T("BLACK"));
|
|
|
|
|
|
|
|
this->SetLayoutCache (wxSTC_CACHE_NONE);
|
|
|
|
this->SetUseHorizontalScrollBar(1);
|
|
|
|
|
|
|
|
// set spaces and indention
|
|
|
|
this->SetTabWidth(4);
|
|
|
|
this->SetUseTabs(true);
|
|
|
|
this->SetTabIndents(true);
|
|
|
|
this->SetBackSpaceUnIndents(true);
|
|
|
|
|
2015-04-15 09:47:20 -04:00
|
|
|
if (keywords_array.isArray())
|
|
|
|
{
|
2015-04-15 12:17:25 -04:00
|
|
|
for(int i = 0; i < keywords_array.size(); i++)
|
|
|
|
{
|
|
|
|
this->SetKeyWords(i, keywords_array[i].asString());
|
|
|
|
}
|
2015-04-15 09:47:20 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-04-16 17:03:27 -04:00
|
|
|
string typeMap[] = {"null", "int", "unsigned int", "double", "string", "boolean", "array", "object"};
|
2015-04-17 16:55:48 -04:00
|
|
|
stringstream output;
|
2015-04-16 17:03:27 -04:00
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
output << "current lang is:" << lang << endl;
|
|
|
|
output << "keywords array is not an array" << endl;
|
|
|
|
output << "keyword array is a " << typeMap[keywords_array.type()] << endl;
|
|
|
|
|
|
|
|
wxLogDebug(output.str().c_str());
|
2015-04-15 09:47:20 -04:00
|
|
|
}
|
2015-04-17 16:55:48 -04:00
|
|
|
}
|
2015-04-10 15:11:15 -04:00
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
/**
|
|
|
|
* Check file path and open the selected file
|
|
|
|
*
|
|
|
|
* @param wxString filePath
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
bool EditPane::Load(wxString filePath)
|
|
|
|
{
|
|
|
|
this->fileName.Assign(filePath);
|
|
|
|
|
|
|
|
if (this->FileReadable())
|
|
|
|
{
|
|
|
|
this->Highlight(filePath);
|
|
|
|
bool didLoad = this->LoadFile(filePath);
|
|
|
|
|
|
|
|
// @TODO Toggle controls based on write permission
|
|
|
|
|
|
|
|
return didLoad;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2015-04-10 15:11:15 -04:00
|
|
|
}
|
2015-04-16 17:03:27 -04:00
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
/**
|
|
|
|
* Determine the format of the current file by
|
|
|
|
* matching its extension against the patterns
|
|
|
|
* in the configuration files
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2015-04-16 17:03:27 -04:00
|
|
|
string EditPane::GetLangByFile()
|
|
|
|
{
|
|
|
|
JsonValue langList = config->GetRoot();
|
|
|
|
JsonValue::iterator it;
|
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
wxString curr_file = this->fileName.GetFullName();
|
2015-04-16 17:03:27 -04:00
|
|
|
|
|
|
|
// Loop through each language to find a matching file pattern
|
|
|
|
for (it = langList.begin(); it != langList.end(); ++it)
|
|
|
|
{
|
|
|
|
string lang = it.key().asString();
|
|
|
|
|
|
|
|
// Parse the file pattern
|
|
|
|
wxString file_pattern((*it)["file_pattern"].asString());
|
|
|
|
|
|
|
|
file_pattern.Lower();
|
|
|
|
|
|
|
|
while ( ! file_pattern.empty())
|
|
|
|
{
|
|
|
|
wxString cur = file_pattern.BeforeFirst(';');
|
|
|
|
if (
|
|
|
|
(cur == curr_file) ||
|
|
|
|
(cur == (curr_file.BeforeLast('.') + _T(".*"))) ||
|
|
|
|
(cur == (_T("*.") + curr_file.AfterLast('.')))
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return lang;
|
|
|
|
}
|
|
|
|
file_pattern = file_pattern.AfterFirst(';');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
bool EditPane::SaveFile()
|
|
|
|
{
|
2015-04-17 16:55:48 -04:00
|
|
|
wxString fname;
|
|
|
|
|
|
|
|
if ( ! this->fileName.IsOk())
|
2015-04-16 17:03:27 -04:00
|
|
|
{
|
|
|
|
wxFileDialog dlg (
|
|
|
|
this,
|
|
|
|
_T("Save file"),
|
|
|
|
wxEmptyString,
|
|
|
|
wxEmptyString,
|
|
|
|
TYRO_FILE_SAVE_WILDCARDS,
|
|
|
|
wxFD_SAVE | wxFD_OVERWRITE_PROMPT
|
|
|
|
);
|
|
|
|
|
|
|
|
if (dlg.ShowModal() != wxID_OK) return false;
|
2015-04-17 16:55:48 -04:00
|
|
|
fname = dlg.GetPath();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fname = this->fileName.GetFullPath();
|
2015-04-16 17:03:27 -04:00
|
|
|
}
|
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
const wxString cfname(fname);
|
|
|
|
|
|
|
|
return this->SaveFile(cfname);
|
2015-04-16 17:03:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool EditPane::SaveFile(const wxString &filename)
|
|
|
|
{
|
|
|
|
if ( ! this->IsModified()) return true;
|
2015-04-17 12:48:00 -04:00
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
this->fileName.Assign(filename);
|
|
|
|
|
|
|
|
// Check file permissions
|
|
|
|
if (this->FileWritable())
|
|
|
|
{
|
|
|
|
this->SetSavePoint();
|
|
|
|
return wxStyledTextCtrl::SaveFile(filename);
|
|
|
|
}
|
2015-04-17 12:48:00 -04:00
|
|
|
|
2015-04-17 16:55:48 -04:00
|
|
|
return false;
|
2015-04-16 17:03:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool EditPane::IsModified()
|
|
|
|
{
|
2015-04-17 16:55:48 -04:00
|
|
|
return this->GetModify();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check that the current file can be opened
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
bool EditPane::FileReadable()
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
this->fileName.IsOk() &&
|
|
|
|
this->fileName.Exists() &&
|
|
|
|
this->fileName.IsFileReadable()
|
|
|
|
) return true;
|
|
|
|
|
|
|
|
// Hmm...well, let's give an error
|
|
|
|
wxMessageDialog errDlg(
|
|
|
|
this,
|
|
|
|
TYRO_OPEN_ERROR,
|
|
|
|
TYRO_OPEN_ERROR_CAPTION,
|
|
|
|
wxOK | wxICON_ERROR | wxCENTER
|
|
|
|
);
|
|
|
|
errDlg.ShowModal();
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check that the current file can be saved
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
bool EditPane::FileWritable()
|
|
|
|
{
|
|
|
|
// Lets see...can we write somewhere?
|
|
|
|
if (
|
|
|
|
this->fileName.IsOk() &&
|
|
|
|
((this->fileName.Exists() && this->fileName.IsFileWritable()) ||
|
|
|
|
(( ! this->fileName.Exists()) && this->fileName.IsDirWritable()))
|
|
|
|
) return true;
|
|
|
|
|
|
|
|
// Hmm...well, let's give an error
|
|
|
|
wxMessageDialog errDlg(
|
|
|
|
this,
|
|
|
|
TYRO_SAVE_ERROR,
|
|
|
|
TYRO_SAVE_ERROR_CAPTION,
|
|
|
|
wxOK | wxICON_ERROR | wxCENTER
|
|
|
|
);
|
|
|
|
errDlg.ShowModal();
|
|
|
|
|
|
|
|
return false;
|
2015-04-16 17:03:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void EditPane::BindEvents()
|
|
|
|
{
|
|
|
|
Bind(wxEVT_STC_MARGINCLICK, &EditPane::OnMarginClick, this, wxID_ANY);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditPane::OnMarginClick(wxStyledTextEvent& event)
|
|
|
|
{
|
|
|
|
if (event.GetMargin() == MARGIN_FOLD) {
|
|
|
|
int lineClick = this->LineFromPosition (event.GetPosition());
|
|
|
|
int levelClick = this->GetFoldLevel (lineClick);
|
|
|
|
if ((levelClick & wxSTC_FOLDLEVELHEADERFLAG) > 0) {
|
|
|
|
this->ToggleFold (lineClick);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|