#pragma once /* editor.hpp Редактор */ /* Copyright (c) 1996 Eugene Roshal Copyright (c) 2000 Far Group All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. The name of the authors may not be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "scrobj.hpp" #include #include "poscache.hpp" #include "bitflags.hpp" #include "config.hpp" #include "DList.hpp" #include "noncopyable.hpp" #include "FARString.hpp" class FileEditor; class KeyBar; struct InternalEditorBookMark { DWORD64 Line[POSCACHE_BOOKMARK_COUNT]; DWORD64 Cursor[POSCACHE_BOOKMARK_COUNT]; DWORD64 ScreenLine[POSCACHE_BOOKMARK_COUNT]; DWORD64 LeftPos[POSCACHE_BOOKMARK_COUNT]; }; struct InternalEditorStackBookMark { DWORD Line; DWORD Cursor; DWORD ScreenLine; DWORD LeftPos; InternalEditorStackBookMark *prev, *next; }; struct EditorCacheParams { int Line; int LinePos; int ScreenLine; int LeftPos; int CodePage; InternalEditorBookMark SavePos; }; struct EditorUndoData { int Type; int StrPos; int StrNum; wchar_t EOL[10]; int Length; wchar_t *Str; EditorUndoData() { memset(this, 0, sizeof(*this)); } ~EditorUndoData() { if (Str) { delete[] Str; } } void SetData(int Type,const wchar_t *Str,const wchar_t *Eol,int StrNum,int StrPos,int Length=-1) { if (Length == -1 && Str) Length=(int)StrLength(Str); this->Type=Type; this->StrPos=StrPos; this->StrNum=StrNum; this->Length=Length; far_wcsncpy(EOL,Eol?Eol:L"",ARRAYSIZE(EOL)-1); if (this->Str) { delete[] this->Str; } if (Str) { this->Str=new wchar_t[Length+1]; if (this->Str) wmemmove(this->Str,Str,Length); } else this->Str=nullptr; } }; // Младший байт (маска 0xFF) юзается классом ScreenObject!!! enum FLAGS_CLASS_EDITOR { FEDITOR_MODIFIED = 0x00000200, FEDITOR_JUSTMODIFIED = 0x00000400, // 10.08.2000 skv: need to send EE_REDRAW 2. // set to 1 by TextChanged, no matter what // is value of State. FEDITOR_MARKINGBLOCK = 0x00000800, FEDITOR_MARKINGVBLOCK = 0x00001000, FEDITOR_WASCHANGED = 0x00002000, FEDITOR_OVERTYPE = 0x00004000, FEDITOR_NEWUNDO = 0x00010000, FEDITOR_UNDOSAVEPOSLOST = 0x00020000, FEDITOR_DISABLEUNDO = 0x00040000, // возможно процесс Undo уже идет? FEDITOR_LOCKMODE = 0x00080000, FEDITOR_CURPOSCHANGEDBYPLUGIN = 0x00100000, // TRUE, если позиция в редакторе была изменена // плагином (ECTL_SETPOSITION) FEDITOR_ISRESIZEDCONSOLE = 0x00800000, FEDITOR_PROCESSCTRLQ = 0x02000000, // нажата Ctrl-Q и идет процесс вставки кода символа FEDITOR_DIALOGMEMOEDIT = 0x80000000, // Editor используется в диалоге в качестве DI_MEMOEDIT }; class Edit; class Editor:public ScreenObject { friend class DlgEdit; friend class FileEditor; private: /* $ 04.11.2003 SKV на любом выходе если была нажата кнопка выделения, и она его "сняла" (сделала 0-й ширины), то его надо убрать. */ class EditorBlockGuard:public NonCopyable { Editor& ed; void (Editor::*method)(); bool needCheckUnmark; public: void SetNeedCheckUnmark(bool State) {needCheckUnmark=State;} EditorBlockGuard(Editor& ed,void (Editor::*method)()):ed(ed),method(method),needCheckUnmark(false) { } ~EditorBlockGuard() { if (needCheckUnmark)(ed.*method)(); } }; DList UndoData; EditorUndoData *UndoPos; EditorUndoData *UndoSavePos; int UndoSkipLevel; int LastChangeStrPos; int NumLastLine; int NumLine; /* $ 26.02.2001 IS Сюда запомним размер табуляции и в дальнейшем будем использовать его, а не Opt.TabSize */ EditorOptions EdOpt; int Pasting; wchar_t GlobalEOL[10]; // работа с блоками из макросов (MCODE_F_EDITOR_SEL) Edit *MBlockStart; int MBlockStartX; Edit *BlockStart; int BlockStartLine; Edit *VBlockStart; int VBlockX; int VBlockSizeX; int VBlockY; int VBlockSizeY; int MaxRightPos; int XX2; //scrollbar FARString strLastSearchStr; /* $ 30.07.2000 KM Новая переменная для поиска "Whole words" */ int LastSearchCase,LastSearchWholeWords,LastSearchReverse,LastSearchSelFound,LastSearchRegexp; UINT m_codepage; //BUGBUG int StartLine; int StartChar; InternalEditorBookMark SavePos; InternalEditorStackBookMark *StackPos; BOOL NewStackPos; int EditorID; FileEditor *HostFileEditor; Edit *TopList; Edit *EndList; Edit *TopScreen; Edit *CurLine; Edit *LastGetLine; int LastGetLineNumber; private: virtual void DisplayObject(); void ShowEditor(int CurLineOnly); void DeleteString(Edit *DelPtr,int LineNumber,int DeleteLast,int UndoLine); void InsertString(); void Up(); void Down(); void ScrollDown(); void ScrollUp(); BOOL Search(int Next); void GoToLine(int Line); void GoToPosition(); void TextChanged(int State); int CalcDistance(Edit *From, Edit *To,int MaxDist); void Paste(const wchar_t *Src=nullptr); void Copy(int Append); void DeleteBlock(); void UnmarkBlock(); void UnmarkEmptyBlock(); void UnmarkMacroBlock(); void ProcessPasteEvent(); void AddUndoData(int Type,const wchar_t *Str=nullptr,const wchar_t *Eol=nullptr,int StrNum=0,int StrPos=0,int Length=-1); void Undo(int redo); void SelectAll(); //void SetStringsTable(); void BlockLeft(); void BlockRight(); void DeleteVBlock(); void VCopy(int Append); void VPaste(wchar_t *ClipText); void VBlockShift(int Left); Edit* GetStringByNumber(int DestLine); static void EditorShowMsg(const wchar_t *Title,const wchar_t *Msg, const wchar_t* Name,int Percent); int SetBookmark(DWORD Pos); int GotoBookmark(DWORD Pos); int ClearStackBookmarks(); int DeleteStackBookmark(InternalEditorStackBookMark *sb_delete); int RestoreStackBookmark(); int AddStackBookmark(BOOL blNewPos=TRUE); InternalEditorStackBookMark* PointerToFirstStackBookmark(int *piCount=nullptr); InternalEditorStackBookMark* PointerToLastStackBookmark(int *piCount=nullptr); InternalEditorStackBookMark* PointerToStackBookmark(int iIdx); int BackStackBookmark(); int PrevStackBookmark(); int NextStackBookmark(); int LastStackBookmark(); int GotoStackBookmark(int iIdx); int PushStackBookMark(); int PopStackBookMark(); int CurrentStackBookmarkIdx(); int GetStackBookmark(int iIdx,EditorBookMarks *Param); int GetStackBookmarks(EditorBookMarks *Param); int BlockStart2NumLine(int *Pos); int BlockEnd2NumLine(int *Pos); bool CheckLine(Edit* line); wchar_t *Block2Text(wchar_t *ptrInitData); wchar_t *VBlock2Text(wchar_t *ptrInitData); public: Editor(ScreenObject *pOwner=nullptr,bool DialogUsed=false); virtual ~Editor(); public: void SetCacheParams(EditorCacheParams *pp); void GetCacheParams(EditorCacheParams *pp); bool SetCodePage(UINT codepage); //BUGBUG UINT GetCodePage(); //BUGBUG int SetRawData(const wchar_t *SrcBuf,int SizeSrcBuf,int TextFormat); // преобразование из буфера в список int GetRawData(wchar_t **DestBuf,int& SizeDestBuf,int TextFormat=0); // преобразование из списка в буфер virtual int ProcessKey(int Key); virtual int ProcessMouse(MOUSE_EVENT_RECORD *MouseEvent); virtual int64_t VMProcess(int OpCode,void *vParam=nullptr,int64_t iParam=0); void KeepInitParameters(); void SetStartPos(int LineNum,int CharNum); BOOL IsFileModified() const; BOOL IsFileChanged() const; void SetTitle(const wchar_t *Title); long GetCurPos(); int EditorControl(int Command,void *Param); void SetHostFileEditor(FileEditor *Editor) {HostFileEditor=Editor;}; static void SetReplaceMode(int Mode); FileEditor *GetHostFileEditor() {return HostFileEditor;}; void PrepareResizedConsole() {Flags.Set(FEDITOR_ISRESIZEDCONSOLE);} void SetTabSize(int NewSize); int GetTabSize() const {return EdOpt.TabSize; } void SetConvertTabs(int NewMode); int GetConvertTabs() const {return EdOpt.ExpandTabs; } void SetDelRemovesBlocks(int NewMode); int GetDelRemovesBlocks() const {return EdOpt.DelRemovesBlocks; } void SetPersistentBlocks(int NewMode); int GetPersistentBlocks() const {return EdOpt.PersistentBlocks; } void SetAutoIndent(int NewMode) { EdOpt.AutoIndent=NewMode; } int GetAutoIndent() const {return EdOpt.AutoIndent; } void SetAutoDetectCodePage(int NewMode) { EdOpt.AutoDetectCodePage=NewMode; } int GetAutoDetectCodePage() const {return EdOpt.AutoDetectCodePage; } void SetCursorBeyondEOL(int NewMode); int GetCursorBeyondEOL() const {return EdOpt.CursorBeyondEOL; } void SetBSLikeDel(int NewMode) { EdOpt.BSLikeDel=NewMode; } int GetBSLikeDel() const {return EdOpt.BSLikeDel; } void SetCharCodeBase(int NewMode) { EdOpt.CharCodeBase=NewMode%3; } int GetCharCodeBase() const {return EdOpt.CharCodeBase; } void SetReadOnlyLock(int NewMode) { EdOpt.ReadOnlyLock=NewMode&3; } int GetReadOnlyLock() const {return EdOpt.ReadOnlyLock; } void SetShowScrollBar(int NewMode) {EdOpt.ShowScrollBar=NewMode;} void SetSearchPickUpWord(int NewMode) {EdOpt.SearchPickUpWord=NewMode;} void SetWordDiv(const wchar_t *WordDiv) { EdOpt.strWordDiv = WordDiv; } const wchar_t *GetWordDiv() { return EdOpt.strWordDiv; } void SetShowWhiteSpace(int NewMode); void GetSavePosMode(int &SavePos, int &SaveShortPos); // передавайте в качестве значения параметра "-1" для параметра, // который не нужно менять void SetSavePosMode(int SavePos, int SaveShortPos); void GetRowCol(const wchar_t *argv,int *row,int *col); int GetLineCurPos(); void BeginVBlockMarking(); void AdjustVBlock(int PrevX); void Xlat(); static void PR_EditorShowMsg(); void FreeAllocatedData(bool FreeUndo=true); Edit *CreateString(const wchar_t *lpwszStr, int nLength); Edit *InsertString(const wchar_t *lpwszStr, int nLength, Edit *pAfter = nullptr, int AfterLineNumber=-1); void SetDialogParent(DWORD Sets); void SetReadOnly(int NewReadOnly) {Flags.Change(FEDITOR_LOCKMODE,NewReadOnly);}; int GetReadOnly() {return Flags.Check(FEDITOR_LOCKMODE);}; void SetOvertypeMode(int Mode); int GetOvertypeMode(); void SetEditBeyondEnd(int Mode); void SetClearFlag(int Flag); int GetClearFlag(); int GetCurCol(); int GetCurRow() {return NumLine;} void SetCurPos(int NewCol, int NewRow=-1); void SetCursorType(bool Visible, DWORD Size); void GetCursorType(bool& Visible, DWORD& Size); void SetObjectColor(int Color,int SelColor,int ColorUnChanged); void DrawScrollbar(); };