代码拉取完成,页面将自动刷新
unit wMainDml;
{$MODE Delphi}
{$WARN 5057 off : Local variable "$1" does not seem to be initialized}
{$WARN 4105 off : Implicit string type conversion with potential data loss from "$1" to "$2"}
{$define EZDML_CHATGPT}
{$define USE_MSSQL}
{$ifdef WIN32}
{$undef EZDML_CHATGPT}
{$endif}
{$ifdef EZDML_LITE}
{$undef EZDML_CHATGPT}
{$ifdef DARWIN}
{$undef USE_MSSQL}
{$endif}
{$endif}
interface
uses
LCLIntf, LCLType, LMessages, Messages, SysUtils, Variants, Classes,
Graphics, Controls, Forms,
Dialogs, Menus, ComCtrls, StdCtrls, ExtCtrls, WindowFuncs, {XPMan,}
uFrameCtTableDef, CtMetaTable, CTMetaData, CtObjSerialer, CtObjXmlSerial, wDmlHelp,
{$ifndef EZDML_LITE}
BESENCharset,
DmlJsScript,
{$endif}
uWaitWnd, ActnList, StdActns, Buttons, FileUtil, CtObjJsonSerial;
const
WMZ_CUSTCMD = WM_USER + $1001;
type
{ TfrmMainDml }
TfrmMainDml = class(TForm)
actEditGlobalScript: TAction;
actImportFile: TAction;
actCharCodeTool: TAction;
actGenerateLastCode: TAction;
actHttpServer: TAction;
actCheckUpdates: TAction;
actFullScreen: TAction;
actGenerateTestData: TAction;
actImportExcel: TAction;
actChatGPT: TAction;
actImportDDLSql: TAction;
actOpenUrl: TAction;
actShareFile: TAction;
actNewAppWin: TAction;
actSaveToDb: TAction;
actLoadFromDb: TAction;
actRefresh: TAction;
actShowHideList: TAction;
actToggleTableView: TAction;
actShowDescText: TAction;
actSettings: TAction;
ImageListSwitchOnOff: TImageList;
lbNewVerInfo: TLabel;
MainMenu1: TMainMenu;
MenuItem1: TMenuItem;
MenuItem2: TMenuItem;
MenuItem3: TMenuItem;
MN_ImportDDLSql: TMenuItem;
MNAI_GenFKLinks: TMenuItem;
MNAI_GenComments: TMenuItem;
MNAI_GenFields: TMenuItem;
MNAI_GenTables: TMenuItem;
MNAI_GenNewModel: TMenuItem;
MNOpenURL: TMenuItem;
MnShareFile: TMenuItem;
MnNewAppWin: TMenuItem;
MNChatGPT1: TMenuItem;
MNImportExcel: TMenuItem;
MNImportFile: TMenuItem;
MNSaveToDb: TMenuItem;
MNLoadFromDb: TMenuItem;
MN_Refresh: TMenuItem;
MnGenerateTestData: TMenuItem;
MN_ShowHideList: TMenuItem;
MN_ToggleTableView: TMenuItem;
MN_FullScreen: TMenuItem;
MN_CheckUpdates: TMenuItem;
MN_HttpServer: TMenuItem;
MnGenerateLastCode: TMenuItem;
MN_Settings: TMenuItem;
MN_FindHex: TMenuItem;
MN_editGlobalScript: TMenuItem;
PanelNewVerHint: TPanel;
Shape1: TShape;
StatusBar1: TStatusBar;
MnOpenFile: TMenuItem;
MnSaveFile: TMenuItem;
MnExit: TMenuItem;
MnQuickStart: TMenuItem;
TimerDelayCmd: TTimer;
TimerInit: TTimer;
OpenDialog1: TOpenDialog;
SaveDialog1: TSaveDialog;
MnImportDatabase: TMenuItem;
MnGenerateDatabase: TMenuItem;
MnNewFile: TMenuItem;
MNNewTable: TMenuItem;
Mn_File: TMenuItem;
N2: TMenuItem;
MnExitWithoutSave: TMenuItem;
ActionList1: TActionList;
EditSelectAll1: TEditSelectAll;
MNSaveFileAs: TMenuItem;
MN_Model: TMenuItem;
Mn_Help: TMenuItem;
Mn_About: TMenuItem;
N1: TMenuItem;
MnNewModel: TMenuItem;
MnTogglePhyView: TMenuItem;
MN_ColorStyles: TMenuItem;
N3: TMenuItem;
MN_ExportXls: TMenuItem;
mn_EzdmlHomePage: TMenuItem;
MN_SearchFields: TMenuItem;
MN_Recentfiles: TMenuItem;
actOpenLastFile1: TAction;
MnOpenLastFile1A: TMenuItem;
MN_EditMyDict: TMenuItem;
MnTools1: TMenuItem;
MnBackupDatabase: TMenuItem;
MnRestoreDatabase: TMenuItem;
MNSqlTool: TMenuItem;
TimerAutoSave: TTimer;
MNShowFileInExplorer: TMenuItem;
N4: TMenuItem;
MN_EditINIfile: TMenuItem;
MN_ExecScript: TMenuItem;
MNShowTemprFile: TMenuItem;
MN_BrowseScripts: TMenuItem;
MnGenerateCode: TMenuItem;
N5: TMenuItem;
MnOpenCustomToolFolder: TMenuItem;
actGoTbFilter: TAction;
actNewFile: TAction;
actOpenFile: TAction;
actSaveFile: TAction;
actSaveFileAs: TAction;
actShowFileInExplorer: TAction;
actShowTmprFile: TAction;
actExitWithoutSave: TAction;
actExit: TAction;
actNewTable: TAction;
actNewModel: TAction;
actImportDatabase: TAction;
actGenerateDatabase: TAction;
actGenerateCode: TAction;
actTogglePhyView: TAction;
actModelOptions: TAction;
actExportModel: TAction;
actExecScript: TAction;
actFindObjects: TAction;
actEditSettingFile: TAction;
actEditMyDict: TAction;
actBrowseScripts: TAction;
actBackupDatabase: TAction;
actRestoreDatabase: TAction;
actSqlTool: TAction;
actBrowseCustomTools: TAction;
actQuickStart: TAction;
actEzdmlHomePage: TAction;
actAboutEzdml: TAction;
OpenDialogImp: TOpenDialog;
procedure actCharCodeToolExecute(Sender: TObject);
procedure actChatGPTExecute(Sender: TObject);
procedure actCheckUpdatesExecute(Sender: TObject);
procedure actEditGlobalScriptExecute(Sender: TObject);
procedure actFullScreenExecute(Sender: TObject);
procedure actGenerateLastCodeExecute(Sender: TObject);
procedure actGenerateTestDataExecute(Sender: TObject);
procedure actHttpServerExecute(Sender: TObject);
procedure actImportDDLSqlExecute(Sender: TObject);
procedure actImportExcelExecute(Sender: TObject);
procedure actImportFileExecute(Sender: TObject);
procedure actLoadFromDbExecute(Sender: TObject);
procedure actNewAppWinExecute(Sender: TObject);
procedure actOpenUrlExecute(Sender: TObject);
procedure actOpenLastFile1Execute(Sender: TObject);
procedure actRefreshExecute(Sender: TObject);
procedure actSaveToDbExecute(Sender: TObject);
procedure actSettingsExecute(Sender: TObject);
procedure actShareFileExecute(Sender: TObject);
procedure actShowDescTextExecute(Sender: TObject);
procedure actShowHideListExecute(Sender: TObject);
procedure actToggleTableViewExecute(Sender: TObject);
procedure FormCloseQuery(Sender: TObject; var CanClose: boolean);
procedure FormDropFiles(Sender: TObject; const FileNames: array of string);
procedure lbNewVerInfoClick(Sender: TObject);
procedure MNAI_GenNewModelClick(Sender: TObject);
procedure Shape1MouseUp(Sender: TObject; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer);
procedure TimerDelayCmdTimer(Sender: TObject);
procedure TimerInitTimer(Sender: TObject);
procedure FormDestroy(Sender: TObject);
procedure FormCreate(Sender: TObject);
procedure TimerAutoSaveTimer(Sender: TObject);
procedure actGoTbFilterExecute(Sender: TObject);
procedure actNewFileExecute(Sender: TObject);
procedure actOpenFileExecute(Sender: TObject);
procedure actSaveFileExecute(Sender: TObject);
procedure actSaveFileAsExecute(Sender: TObject);
procedure actShowFileInExplorerExecute(Sender: TObject);
procedure actShowTmprFileExecute(Sender: TObject);
procedure actExitWithoutSaveExecute(Sender: TObject);
procedure actExitExecute(Sender: TObject);
procedure actNewTableExecute(Sender: TObject);
procedure actNewModelExecute(Sender: TObject);
procedure actImportDatabaseExecute(Sender: TObject);
procedure actGenerateDatabaseExecute(Sender: TObject);
procedure actGenerateCodeExecute(Sender: TObject);
procedure actTogglePhyViewExecute(Sender: TObject);
procedure actModelOptionsExecute(Sender: TObject);
procedure actExportModelExecute(Sender: TObject);
procedure actExecScriptExecute(Sender: TObject);
procedure actFindObjectsExecute(Sender: TObject);
procedure actEditSettingFileExecute(Sender: TObject);
procedure actEditMyDictExecute(Sender: TObject);
procedure actBrowseScriptsExecute(Sender: TObject);
procedure actBackupDatabaseExecute(Sender: TObject);
procedure actRestoreDatabaseExecute(Sender: TObject);
procedure actSqlToolExecute(Sender: TObject);
procedure actBrowseCustomToolsExecute(Sender: TObject);
procedure actQuickStartExecute(Sender: TObject);
procedure actEzdmlHomePageExecute(Sender: TObject);
procedure actAboutEzdmlExecute(Sender: TObject);
private
{ Private declarations }
FFrameCtTableDef: TFrameCtTableDef;
FCtDataModelList: TCtDataModelGraphList;
FCurFileName: string;
FCurFileSize: integer;
FMainSplitterPos: integer;
FCurFileDate: TDateTime;
FFileDbConnectOk: Boolean;
FfrmMetaImport: TForm;
FfrmHttpServer: TForm;
FFindHexDlg: TForm;
FProgressAll: integer;
FProgressCur: integer;
FWaitWnd: TfrmWaitWnd;
FOrginalCaption: string;
FGlobeOpeningFile: string;
FRecentFiles: TStringList;
FReservedToolsMenuCount: integer;
FCustomTools: TStringList;
FAutoSaveMinutes: integer;
FAutoSaveCounter: integer;
FIsAutoSaving: boolean;
FSaveTempFileOnExit: boolean;
FGlobalScriptor: TObject;
FFileLockMutex: TCtMutex;
FFileWorking: boolean;
FStartMaximized: Boolean;
FCheckingFileDate: boolean;
FFullScrnSaveBound: TRect;
procedure _OnDMLObjProgress(Sender: TObject; const Prompt: string;
Cur, All: integer; var bContinue: boolean);
procedure _OnRecentFileClick(Sender: TObject);
procedure _OnAppActivate(Sender: TObject);
procedure _OnDbFileMemoChanged(Sender: TObject; fn: string);
procedure PromptOpenFile(fn: string; bDisableTmpFiles: boolean = False);
procedure LoadFromFile(fn: string);
procedure RememberFileDateSize;
procedure ImportFromFile(fn: string); //导入文件
procedure SaveToFile(fn: string);
procedure PromptSaveFile;
procedure CheckCaption;
procedure LoadFromDbFile(fn: string);
function CheckDbFileState(fn: string; bForce: Boolean): Integer;
procedure LoadIni;
procedure SaveIni;
procedure SetRecentFile(fn: string);
procedure RemoveRecentFile(fn: string);
procedure RecreateRecentMn;
procedure TryLockFile(fn: string; bAsk: boolean = True);
procedure ReCreateCustomToolsMenu;
procedure _OnCustomToolsClick(Sender: TObject);
procedure CallAI(Act: Integer);
function CheckCurFileDateSizeChanged: boolean;
function IsTmpFile(fn: string): boolean;
function IsDbFile(fn: string): boolean;
function IsDbHistFile(fn: string): boolean;
function GetStatusPanelFileName(fn: string): string;
function GetTmpDirForFile(fn: string): string;
function ExtractDmlFileDir(fn: string): string;
function ExtractDmlFileName(fn: string): string;
function GetFastTmpFileName(fn: string): string; //快速加载用的临时文件名
function GetLastTmpFileName(fn: string): string; //最后一次的临时文件名
function GetNewTmpFileName(fn: string): string;
function SaveDMLToTmpFile: string;
procedure SaveDMLFastTmpFile(bForceSaveHuge: Boolean=False);
function TryLoadFromTmpFile(sfn: string): boolean;
procedure CheckReloadGlobalScript;
procedure CheckForUpdates(bForceNow: boolean);
procedure CheckShowNewVersionInfo(bForceNow: boolean);
function GetDmlFileDate(fn: string; var vFileDate: TDateTime): boolean;
function GetDmlFileDateAndSize(fn: string; var vFileSize: Integer; var vFileDate: TDateTime): boolean;
protected
procedure CreateWnd; override;
procedure _WMZ_CUSTCMD(var msg: TMessage); message WMZ_CUSTCMD;
public
{ Public declarations }
function IsShortcut(var Message: TLMKey): boolean; override;
procedure SetStatusBarMsg(msg: string; tp: integer = -1);
procedure ExecDmlScript(fn: string);
end;
procedure CheckAppStart;
function EzdmlExecAppCmd(Cmd, param1, param2: string): string;
function EzdmlCreateCtObjSerialer(fn: string; bWriteMode: boolean): TCtObjSerialer;
function IsSameFileContent(// 比较两个文件是否相等
mFileName1, mFileName2: string // 两个文件
): boolean;
var
frmMainDml: TfrmMainDml;
implementation
uses
uFormImpTable, uFormGenSql, uFormCtDML, CtMetaOracleDb, uFormEzdmlDbFile,
{$ifndef EZDML_LITE}
CtMetaPdmImporter, DmlPasScript, DmlGlobalPasScript, ide_editor, uFormGenCode,
uFormHttpSvr, FindHexDlg, wExcelImp, DmlScriptControl, uFormGenData, CtTestDataGen,
wDDLSqlImp,
{$else}
DmlGlobalPasScriptLite, DmlPasScriptLite,
{$endif}
{$ifdef EZDML_CHATGPT}uFormChatGPT, ChatGptIntf,{$endif}
CtMetaOdbcDb, NetUtil, PvtInput, AESCrypt, MD5, Base64,
ocidyn, mysql57dyn, sqlite3dyn, CtSysInfo, wShareFile, uFormOnlineFile,
postgres3dyn,
ezdmlstrs, dmlstrs, DMLObjs, IniFiles, AutoNameCapitalize, uDMLSqlEditor,
wAbout, wSettings, uFormCtTableProp, uFormCtFieldProp,
uJSON, DmlScriptPublic, CtMetaSqliteDb,
uPSComponent, LCLTranslator, uFormCtDbLogon,
{$IFDEF DARWIN} MacOSAll,{$ENDIF}
{$IFDEF USE_MSSQL} CtMetaSqlsvrDb, mssqlconn, dblib, {$ENDIF}
CtMetaMysqlDb, CtMetaPostgreSqlDb, LCLProc, CtMetaHttpDb,
MessageBoxOnTop;
{$R *.lfm}
{$IFDEF DARWIN}
function GetOSLanguageEz: string;
{独立于平台的方法来读取用户界面语言}
var
l, fbl: string;
theLocaleRef: CFLocaleRef;
locale: CFStringRef;
buffer: StringPtr;
bufferSize: CFIndex;
encoding: CFStringEncoding;
success: boolean;
begin
theLocaleRef := CFLocaleCopyCurrent;
locale := CFLocaleGetIdentifier(theLocaleRef);
encoding := 0;
bufferSize := 256;
buffer := new(StringPtr);
success := CFStringGetPascalString(locale, buffer, bufferSize, encoding);
if success then
l := string(buffer^)
else
l := '';
fbl := Copy(l, 1, 2);
dispose(buffer);
Result := fbl;
end;
{$ENDIF}
procedure CheckAppStart;
var
ini: TIniFile;
fn, dir, S: string;
begin
try
if G_CtAppFormHandler = nil then
begin
G_CtAppFormHandler := TCtAppFormHandler.Create;
Screen.AddHandlerFormAdded(G_CtAppFormHandler.ScreenFormAddEvent);
Screen.AddHandlerFormVisibleChanged(G_CtAppFormHandler.ScreenFormVisibleChgEvent);
end;
S := '';
fn := GetConfFileOfApp;
if FileExists(fn) then
begin
ini := TIniFile.Create(fn);
try
S := ini.ReadString('Options', 'LANG', '');
G_AppDefFontName := ini.ReadString('Options', 'AppDefFontName', '');
G_AppDefFontSize := ini.ReadInteger('Options', 'AppDefFontSize', 0);
G_AppFixWidthFontName := ini.ReadString('Options', 'AppFixWidthFontName', '');
G_AppFixWidthFontSize := ini.ReadInteger('Options', 'AppFixWidthFontSize', 0);
G_DmlGraphFontName := ini.ReadString('Options', 'DmlGraphFontName', '');
finally
ini.Free;
end;
end;
dir := '';
{$IFDEF DARWIN}
if S = '' then
S := GetOSLanguageEz;
dir := GetFolderPathOfAppExe('languages');
{$ENDIF}
{$IF FPC_FULLVERSION>30200} //不知哪个版本开始,语言设置的函数改了
S := SetDefaultLang(S, dir); //如果这句编译不过,可改用后面两句
{$ELSE}
SetDefaultLang(S, dir);
S := GetDefaultLang;
{$ENDIF}
SetEzdmlLang(S);
InitCtChnNames;
except
end;
end;
function EzdmlExecAppCmd(Cmd, param1, param2: string): string;
function SaveDmlGraphFile(dmlName, fn: string): string;
var
dml: TCtDataModelGraph;
bBrf: boolean;
begin
if dmlName = '(CUR_DATA_MODEL)' then
with frmMainDml.FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML do
begin
bBrf := DMLGraph.DMLObjs.BriefMode;
DMLGraph.DMLObjs.BriefMode := Pos('(BRIEF)', fn) > 0;
Result := SaveDmlImage(fn);
DMLGraph.DMLObjs.BriefMode := bBrf;
DMLGraph.Refresh;
Exit;
end;
with TfrmCtDML.Create(Application) do
try
dml := TCtDataModelGraph(FGlobeDataModelList.ItemByName(dmlName));
if dml = nil then
dml := FGlobeDataModelList.CurDataModel;
Init(dml, True, True);
FFrameCtDML.DMLGraph.ViewScale := 1;
if Pos('(BRIEF)', fn) > 0 then
FFrameCtDML.DMLGraph.DMLObjs.BriefMode := True;
Result := FFrameCtDML.SaveDmlImage(fn);
finally
Free;
end;
end;
function SaveTableGraphFile(tbName, fn: string): string;
var
tb, tb2: TCtMetaTable;
dml: TCtDataModelGraph;
begin
tb := frmMainDml.FFrameCtTableDef.GetCurTable;
if tbName <> '(CUR_TABLE)' then
tb := FGlobeDataModelList.GetTableOfName(tbName);
if tb = nil then
Exit;
dml := TCtDataModelGraph.Create;
with TfrmCtDML.Create(Application) do
try
tb2 := dml.Tables.NewTableItem();
tb2.AssignFrom(tb);
Init(dml, True, True);
FFrameCtDML.DMLGraph.ViewScale := 1;
Result := FFrameCtDML.SaveDmlImage(fn);
finally
Free;
dml.Free;
end;
end;
begin
Result := '';
if cmd = 'GET_DML_GRAPH_BASE64TEXT' then
begin
Result := SaveDmlGraphFile(param1, '(BASE64TEXT)' + param2);
end;
if cmd = 'SAVE_DML_GRAPH_FILE' then
begin
Result := SaveDmlGraphFile(param1, param2);
end;
if cmd = 'GET_TABLE_GRAPH_BASE64TEXT' then
begin
Result := SaveTableGraphFile(param1, '(BASE64TEXT)' + param2);
end;
if cmd = 'SAVE_TABLE_GRAPH_FILE' then
begin
Result := SaveTableGraphFile(param1, param2);
end;
if cmd = 'GET_PARAM_STR' then
begin
Result := ParamStr(StrToIntDef(param1, 0));
end;
if cmd = 'SET_JSON_UHEX_MODE' then
begin
if StrToIntDef(param1, 0) = 0 then
stringsAsUtf8Encode := True
else
stringsAsUtf8Encode := False;
if stringsAsUtf8Encode then
Result := '0'
else
Result := '1';
end;
end;
function EzdmlCreateCtObjSerialer(fn: string; bWriteMode: boolean): TCtObjSerialer;
var
ext: string;
begin
ext := LowerCase(ExtractFileExt(fn));
if bWriteMode then
begin
if (ext = '.dmx') or (ext = '.xml') then
Result := TCtObjXmlSerialer.Create(fn, fmCreate)
else if (ext = '.dmj') or (ext = '.~dmj') or (ext = '.json') then
begin
Result := TCtObjJsonSerialer.Create(fn, fmCreate);
//TCtObjJsonSerialer(Result).WriteEmptyVals:=True;
end
else
Result := TCtObjFileStream.Create(fn, fmCreate);
end
else
begin
if (ext = '.dmx') or (ext = '.xml') then
Result := TCtObjXmlSerialer.Create(fn, fmOpenRead or fmShareDenyNone)
else if (ext = '.dmj') or (ext = '.~dmj') or (ext = '.json') then
Result := TCtObjJsonSerialer.Create(fn, fmOpenRead or fmShareDenyNone)
else
Result := TCtObjFileStream.Create(fn, fmOpenRead or fmShareDenyNone);
end;
if not bWriteMode then
Result.CurCtVer := 0;
end;
function Ezdml_JsonPropProc(AJsonStr, AName, AValue: string; bRead: boolean): string;
var
js: TJSONObject;
begin
Result := '';
if Trim(AJsonStr) <> '' then
js := TJSONObject.Create(AJsonStr)
else
js := TJSONObject.Create;
try
if bRead then
Result := js.optString(AName)
else
begin
js.put(AName, AValue);
Result := js.toString;
end;
finally
js.Free;
end;
end;
function ADecryptStr(const S, Key,IV: string): string;
function Min(const A, B: Integer): Integer;
begin
if A < B then
Result := A
else
Result := B;
end;
var
SrcStream, TgtStream: TStringStream;
AESKey: TAESKey256;
InitVector: TAESBuffer;
begin
Result := '';
SrcStream := TStringStream.Create(S);
TgtStream := TStringStream.Create('');
try
FillChar(AESKey, SizeOf(AESKey), 0);
Move(PChar(Key)^, AESKey, Min(SizeOf(AESKey), Length(Key)));
Move(PChar(IV)^, InitVector, Min(SizeOf(InitVector), Length(IV)));
DecryptAESStreamCBC(SrcStream, SrcStream.Size - SrcStream.Position, AESKey,InitVector, TgtStream);
Result := TgtStream.DataString;
finally
SrcStream.Free;
TgtStream.Free;
end;
end;
function Ezdml_CheckDecDmlData(data: string): string;
function MakeKey(S: string): string;
begin
while Length(S)<32 do
S:=S+S;
Result := Copy(S,1,32);
end;
function MakeIV(S: string): string;
var
md: TMD5Digest;
begin
md:=MD5String(S);
SetLength(Result, 16);
Move(md, PChar(Result)^, 16);
end;
var
S, str, hd, V, pwd: string;
I, po: Integer;
begin
S := data;
if Copy(S,1,13)='[DMJ_ENC_AES]' then
begin
po := Pos('TITLE=', S);
if po=0 then
raise Exception.Create('ENC TITLE not found');
S:=Copy(S, po+6, Length(S));
po := Pos(#10, S);
if po=0 then
raise Exception.Create('read ENC TITLE error');
V := Trim(Copy(S,1,po-1));
if Pos('%', V)>0 then
V := urlDecode(V);
S:=Copy(S,po+1,Length(S));
str := DecodeStringBase64(S);
for I:=1 to 3 do
begin
if I>1 then
pwd := PvtInputBox(V, srPwdDecryptFailed+' '+srEnterPwd, '*****')
else
pwd := PvtInputBox(V, srEnterPwd, '*****');
if pwd='' then
Abort;
S := ADecryptStr(str,MakeKey(pwd), MakeIV(V));
po := Pos(#10, S);
hd := Trim(Copy(S,1,po-1));
S:=Copy(S,po+1,Length(S));
po :=Pos(#10+hd+#10, S);
S:=Copy(S,1,po-1);
if Copy(S,1,1)<>'{' then
begin
if I>=3 then
raise Exception.Create(srPwdDecryptFailed);
end
else
Break;
end;
end;
Result := S;
end;
function SetFileAges(fn: string; vFileDate: TDateTime): boolean;
var
age: longint;
begin
if not FileExists(fn) then
raise Exception.Create('File not found to set age: '+ fn);
age := DateTimeToFileDate(vFileDate);
Result := (FileSetDate(fn, age) = 0);
{var
Handle, f: THandle;
FindData: TWin32FindData;
LocalFileTime, FileTime: TFileTime;
Age: Integer;
begin
Result := False;
Handle := FindFirstFile(PChar(Fn), FindData);
if Handle <> INVALID_HANDLE_VALUE then
begin
Windows.FindClose(Handle);
f := CreateFile(PChar(fn), GENERIC_READ or GENERIC_WRITE,
FILE_SHARE_READ or FILE_SHARE_DELETE,
nil, OPEN_EXISTING,
FILE_FLAG_BACKUP_SEMANTICS, 0);
//f := FileOpen(fn, fmOpenWrite);
if f = THandle(-1) then
RaiseLastOSError;
if vFileDate > 1 then
begin
Age := DateTimeToFileDate(vFileDate);
if DosDateTimeToFileTime(LongRec(Age).Hi, LongRec(Age).Lo, LocalFileTime) and
LocalFileTimeToFileTime(LocalFileTime, FileTime) then
SetFileTime(f, nil, nil, @FileTime);
end;
if vCreateDate > 1 then
begin
Age := DateTimeToFileDate(vCreateDate);
if DosDateTimeToFileTime(LongRec(Age).Hi, LongRec(Age).Lo, LocalFileTime) and
LocalFileTimeToFileTime(LocalFileTime, FileTime) then
SetFileTime(f, @FileTime, nil, nil);
end;
FileClose(f);
end; }
Result := False;
end;
function ezdml_GetSelectedCtMetaObj: TCtMetaObject;
begin
Result := frmMainDml.FFrameCtTableDef.GetCurObject;
end;
procedure TfrmMainDml.CreateWnd;
begin
inherited;
end;
procedure TfrmMainDml.ExecDmlScript(fn: string);
var
FileTxt, AOutput: TStrings;
S: string;
bUtf8: boolean;
cTb: TCtMetaTable;
begin
cTb := FFrameCtTableDef.GetCurTable;
FileTxt := TStringList.Create;
AOutput := TStringList.Create;
with CreateScriptForFile(fn) do
try
ActiveFile := fn;
FileTxt.LoadFromFile(fn);
S := FileTxt.Text;
bUtf8 := False;
if Length(S) > 3 then
if (Ord(S[1]) = $EF) and (Ord(S[2]) = $BB) and (Ord(S[3]) = $BF) then
begin
S := Copy(S, 4, Length(S));
bUtf8 := True;
end;
if not bUtf8 then
if Pos('UTF-8', UpperCase(S)) >= 0 then
bUtf8 := True;
if bUtf8 then
begin
S := Utf8Decode(S);
end;
Init('DML_SCRIPT', cTb, AOutput, nil);
Exec('DML_SCRIPT', FileTxt.Text);
finally
FileTxt.Free;
AOutput.Free;
Free;
end;
end;
procedure TfrmMainDml.FormCloseQuery(Sender: TObject; var CanClose: boolean);
procedure ReleaseModelList;
begin
Self.Caption := srEzdmlAppTitle + ' - ' + srEzdmlExiting;
Self.SetStatusBarMsg(srEzdmlExiting);
try
if FCtDataModelList.IsHuge then
Screen.Cursor := crAppStart;
try
FCtDataModelList.Clear;
if FCtDataModelList.CurDataModel = nil then
Exit;
//FFrameCtTableDef.Init(FCtDataModelList, False);
FCurFileName := '';
FAutoSaveCounter := 0;
FCurDmlFileName := '';
TryLockFile('');
except
end;
Screen.Cursor := crDefault;
finally
FCurFileName := '';
FAutoSaveCounter := 0;
FCurDmlFileName := '';
end;
end;
var
bCke, bHuge: boolean;
begin
bCke := True;
if GetMetaEditingWin <> nil then
begin
if GetMetaEditingWin = FfrmCtTableProp then
if not FfrmCtTableProp.CheckModified then
begin
FfrmCtTableProp.Close;
bCke := False;
end;
end;
if bCke then
CheckCanEditMeta;
bHuge := FCtDataModelList.IsHuge;
if (FCtDataModelList.TableCount > 0) then
begin
if (IsTmpFile(FCurFileName) or (FCurFileName = '')) then
begin
case Application.MessageBox(PChar(srEzdmlConfirmExit), PChar(Application.Title),
MB_YESNOCANCEL or MB_ICONWARNING) of
idYes:
begin
actSaveFileAs.Execute;
if (IsTmpFile(FCurFileName) or (FCurFileName = '')) then
begin
CanClose := False;
Exit;
end;
end;
idNo:
begin
ReleaseModelList;
end
else
CanClose := False;
Exit;
end;
end
else if bHuge then
begin
PromptSaveFile;
SaveDmlFastTmpFile(True);
end
else if not FSaveTempFileOnExit then
begin
PromptSaveFile;
end;
end;
FFrameCtTableDef.Init(nil, False);
Self.Caption := srEzdmlAppTitle + ' - ' + srEzdmlExiting;
Self.SetStatusBarMsg(srEzdmlExiting);
Self.Refresh;
Application.ProcessMessages;
try
SaveIni;
if not bHuge then
SaveDmlFastTmpFile;
except
end;
if CanClose then
begin
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'CLOSE', '', Self, nil);
end;
end;
if CanClose then
begin
if FMainSplitterPos <> Self.FFrameCtTableDef.PanelCttbTree.Width then
SaveIni;
{$ifndef EZDML_LITE}
if Assigned(scriptIdeEditor) then
FreeAndNil(scriptIdeEditor);
{$endif}
ReleaseModelList;
end;
try
{$ifndef EZDML_LITE}
CheckCaption;
if CanClose then
begin
CheckForUpdates(False);
end;
{$endif}
except
end;
end;
procedure TfrmMainDml.FormDropFiles(Sender: TObject;
const FileNames: array of string);
var
L: integer;
S: string;
begin
L := Length(FileNames);
if L = 0 then
Exit;
S := Trim(FileNames[0]);
FGlobeOpeningFile := S;
PostMessage(Handle, WMZ_CUSTCMD, 1, 0);
end;
procedure TfrmMainDml.lbNewVerInfoClick(Sender: TObject);
var
sNewVer, sUrl, V: string;
ini: TIniFile;
begin
PanelNewVerHint.Hide;
if lbNewVerInfo.Tag <> 1 then
Exit;
ini := TIniFile.Create(GetConfFileOfApp);
try
sNewVer := ini.ReadString('Updates', 'NewVerNum', '');
if sNewVer = '' then
Exit;
ini.WriteString('Updates', 'LastPromptVer', sNewVer);
sUrl := ini.ReadString('Updates', 'NewVerUrl', '');
if sUrl = '' then
Exit;
V := Format(srEzdmlConfirmOpenUrlFmt, [sUrl]);
if Application.MessageBox(PChar(V),
PChar(Application.Title), MB_OKCANCEL or MB_ICONWARNING) <> idOk then
Exit;
CtOpenDoc(PChar(sUrl));
finally
ini.Free;
end;
end;
procedure TfrmMainDml.MNAI_GenNewModelClick(Sender: TObject);
begin
CallAI(TMenuItem(Sender).Tag);
end;
procedure TfrmMainDml.Shape1MouseUp(Sender: TObject; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer);
begin
lbNewVerInfoClick(nil);
end;
procedure TfrmMainDml.TimerDelayCmdTimer(Sender: TObject);
begin
TimerDelayCmd.Enabled := False;
if TimerDelayCmd.Tag = 1 then
begin
if FCtDataModelList.ModelFileConfig.LastModel <> '' then
begin
FFrameCtTableDef.FFrameCtTableList.FocusToModel(FCtDataModelList.ModelFileConfig.LastModel);
FCtDataModelList.ModelFileConfig.LastModel := '';
end;
end;
if TimerDelayCmd.Tag = 11 then
begin
if FCtDataModelList.ModelFileConfig.LastModel <> '' then
begin
FFrameCtTableDef.FFrameCtTableList.FocusToModel(FCtDataModelList.ModelFileConfig.LastModel);
if FFrameCtTableDef.PanelDMLGraph.Visible and
(FFrameCtTableDef.FFrameDMLGraph.MetaTableModel.Name=FCtDataModelList.ModelFileConfig.LastModel) then
TimerDelayCmd.Tag := 12
else
TimerDelayCmd.Tag := 11;
TimerDelayCmd.Enabled := True;
end;
end;
if TimerDelayCmd.Tag = 12 then
begin
if FCtDataModelList.ModelFileConfig.LastModel <> '' then
begin
if FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameDMLGraph.MetaTableModel.Name=FCtDataModelList.ModelFileConfig.LastModel then
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actRearrange.OnExecute(nil);
FCtDataModelList.ModelFileConfig.LastModel := '';
end;
end;
end;
procedure TfrmMainDml.FormCreate(Sender: TObject);
var
db: TCtMetaDatabase;
dpi: Integer;
begin
Randomize;
{$ifndef EZDML_LITE}
G_DmlImageListSwitchOnOff := Self.ImageListSwitchOnOff;
{$endif}
//if LoadNewResourceModule($0409) <> 0 then
//ReinitializeForms();
dpi := Screen.PixelsPerInch;
if dpi = 0 then //由于启动后会自动放大,因此一开始要缩小主窗口
dpi := 96;
Width := Screen.Width * 90 div 100 * 96 div dpi;
Height := Screen.Height * 80 div 100 * 96 div dpi;
Caption := srEzdmlAppTitle;
FOrginalCaption := Caption;
FMainSplitterPos := 150;
AllowDropFiles := True;
FRecentFiles := TStringList.Create;
FCustomTools := TStringList.Create;
FAutoSaveMinutes := 5;
FSaveTempFileOnExit := True;
FCtDataModelList := TCtDataModelGraphList.Create;
FGlobeDataModelList := FCtDataModelList;
FCtDataModelList.OnObjProgress := _OnDMLObjProgress;
FFrameCtTableDef := TFrameCtTableDef.Create(Self);
FFrameCtTableDef.Name := 'FrameCtTableDef';
FFrameCtTableDef.Parent := Self;
FFrameCtTableDef.Align := alClient;
FFrameCtTableDef.Init(FCtDataModelList, False);
FFrameCtTableDef.RefreshProp;
with FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML do
begin
actFileNew.Caption := Self.actNewFile.Caption;
actFileNew.OnExecute := Self.actNewFile.OnExecute;
actFileOpen.Caption := Self.actOpenFile.Caption;
actFileOpen.OnExecute := Self.actOpenFile.OnExecute;
actFileSave.Caption := Self.actSaveFile.Caption;
actFileSave.OnExecute := Self.actSaveFile.OnExecute;
actFullScreen.OnExecute := Self.actFullScreen.OnExecute;
actFullScreen.Visible := True;
actShowHideList.OnExecute := Self.actShowHideList.OnExecute;
actShowHideList.Visible := True;
Porc_OnStatusMsg := Self.SetStatusBarMsg;
end;
frmEzdmlDbFile := TfrmEzdmlDbFile.Create(Self);
frmEzdmlDbFile.Proc_OnDbFileMemoChanged := _OnDbFileMemoChanged;
if GetCtMetaDBReg('ORACLE')^.DbImpl = nil then
begin
db := TCtMetaOracleDb.Create;
GetCtMetaDBReg('ORACLE')^.DbImpl := db;
end;
{$IFDEF USE_MSSQL}
if GetCtMetaDBReg('SQLSERVER')^.DbImpl = nil then
begin
db := TCtMetaSqlsvrDb.Create;
GetCtMetaDBReg('SQLSERVER')^.DbImpl := db;
end;
MsSql_DBLIBDLL := DBLIBDLL;
{$ENDIF}
if GetCtMetaDBReg('MYSQL')^.DbImpl = nil then
begin
db := TCtMetaMysqlDb.Create;
GetCtMetaDBReg('MYSQL')^.DbImpl := db;
end;
if GetCtMetaDBReg('POSTGRESQL')^.DbImpl = nil then
begin
db := TCtMetaPostgreSqlDb.Create;
GetCtMetaDBReg('POSTGRESQL')^.DbImpl := db;
end;
if GetCtMetaDBReg('SQLITE')^.DbImpl = nil then
begin
db := TCtMetaSqliteDb.Create;
GetCtMetaDBReg('SQLITE')^.DbImpl := db;
end;
if GetCtMetaDBReg('ODBC')^.DbImpl = nil then
begin
db := TCtMetaOdbcDb.Create;
GetCtMetaDBReg('ODBC')^.DbImpl := db;
end;
if GetCtMetaDBReg('HTTP_JDBC')^.DbImpl = nil then
begin
db := TCtMetaHttpDb.Create;
GetCtMetaDBReg('HTTP_JDBC')^.DbImpl := db;
end;
{$ifdef EZDML_LITE}
actEditGlobalScript.Visible:=False;
actImportFile.Visible:=False;
actExecScript.Visible:=False;
actBrowseScripts.Visible:=False;
actGenerateCode.Visible:=False;
actCharCodeTool.Visible:=False;
actImportExcel.Visible:=False;
actImportDDLSql.Visible:=False;
actCheckUpdates.Visible:=False;
actGenerateTestData.Visible:=False;
{$endif}
{$ifndef EZDML_CHATGPT}
actChatGPT.Visible := False;
MNChatGPT1.Visible := False;
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actChatGPT.Tag := 1;
{$endif}
LoadIni;
RecreateRecentMn;
ReCreateCustomToolsMenu;
CheckFormScaleDPI(Self);
CheckReloadGlobalScript;
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'CREATE', '', Self, nil);
end;
if ParamStr(1) <> '' then
if GetDmlScriptType(ParamStr(1)) <> '' then
begin
Self.TimerInit.Enabled := False;
try
ExecDmlScript(ParamStr(1));
except
on E: EAbort do
begin
Self.Visible := False;
Application.Terminate;
Application.ShowMainForm := False;
end
else
Application.HandleException(Self);
end;
end;
Application.OnActivate := Self._OnAppActivate;
Application.ExceptionDialog := aedOkMessageBox;
TimerInit.Enabled := True;
end;
procedure TfrmMainDml.LoadFromFile(fn: string);
var
fs: TCtObjSerialer;
begin
if FFileWorking then
Exit;
FFileWorking := True;
try
if Assigned(FWaitWnd) then
raise Exception.Create('wait wnd busy');
if FileExists(fn) then
begin
try
SetStatusBarMsg(Format(srEzdmlOpeningFileFmt, [GetStatusPanelFileName(fn)]));
Self.Refresh;
fs := EzdmlCreateCtObjSerialer(fn, False);
FFrameCtTableDef.IsInitLoading := False;
try
fs.RootName := 'DataModels';
FProgressAll := 0;
FProgressCur := 0;
if IsTmpFile(fn) and (GetDocFileSize(fn) < 1024 * 1024) then
//1MB内的小文件不显示进度条
begin
FWaitWnd := nil;
Screen.Cursor := crAppStart;
end
else
FWaitWnd := TfrmWaitWnd.Create(Self);
try
if Assigned(FWaitWnd) then
FWaitWnd.Init(srEzdmlOpenFile + ' ' + ExtractFileName(fn), srEzdmlOpening,
srEzdmlAbortOpening);
try
FCtDataModelList.Clear;
FCurFileName := '';
CheckCaption;
FFrameCtTableDef.Init(nil, True);
except
end;
FFrameCtTableDef.IsInitLoading := True;
FCtDataModelList.LoadFromSerialer(fs);
finally
Screen.Cursor := crDefault;
if Assigned(FWaitWnd) then
FWaitWnd.Release;
FWaitWnd := nil;
end;
finally
fs.Free;
FFrameCtTableDef.IsInitLoading := False;
FFrameCtTableDef.Init(FCtDataModelList, False);
end;
if FCtDataModelList.ModelFileConfig.LastModel <> '' then
begin
TimerDelayCmd.Tag := 1;
TimerDelayCmd.Enabled := True;
end;
SetStatusBarMsg(GetStatusPanelFileName(fn));
FCurFileName := fn;
Self.RememberFileDateSize;
FAutoSaveCounter := 0;
CheckCaption;
except
on E: Exception do
if not IsTmpFile(fn) then
raise
else
raise Exception.Create(Format(srEzdmlLoadTmpFileFailFmt, [fn, E.message]));
end;
end;
finally
FFileWorking := False;
end;
end;
procedure TfrmMainDml.LoadIni;
var
ini: TIniFile;
I, L, po: integer;
S, V: string;
begin
S := GetAppDefTempPath;
if not DirectoryExists(S) then
ForceDirectories(S);
ini := TIniFile.Create(GetConfFileOfApp);
try
G_AppDefFontName := ini.ReadString('Options', 'AppDefFontName', '');
G_AppDefFontSize := ini.ReadInteger('Options', 'AppDefFontSize', 0);
G_AppFixWidthFontName := ini.ReadString('Options', 'AppFixWidthFontName', '');
G_AppFixWidthFontSize := ini.ReadInteger('Options', 'AppFixWidthFontSize', 0);
G_DmlGraphFontName := ini.ReadString('Options', 'DmlGraphFontName', '');
G_DmlGraphDefScale := ini.ReadString('Options', 'DmlGraphDefScale', '');
I := 0;
FRecentFiles.Clear;
while True do
begin
Inc(I);
S := ini.ReadString('RecentFiles', IntToStr(I), '');
if S = '' then
Break;
FRecentFiles.Add(S);
end;
I := 0;
L := 0;
SetLength(CtCustFieldTypeDefs, L);
while True do
begin
Inc(I);
S := Trim(ini.ReadString('DefaultFieldTypes', IntToStr(I), ''));
if S = '' then
Break;
Inc(L);
SetLength(CtCustFieldTypeDefs, L);
CtCustFieldTypeDefs[L - 1] := S;
end;
SetLength(DML_CustFieldTypeDefs, L);
for I := 0 to L - 1 do
DML_CustFieldTypeDefs[I] := CtCustFieldTypeDefs[I];
I := 0;
L := 0;
SetLength(CtCustFieldTypeList, L);
SetLength(CtCustFieldTypeDefList, L);
while True do
begin
Inc(I);
S := Trim(ini.ReadString('CustFieldTypes', IntToStr(I), ''));
if S = '' then
Break;
Inc(L);
po := Pos(':', S);
if po > 0 then
begin
V := Copy(S, po + 1, Length(S));
S := Copy(S, 1, po - 1);
end
else
begin
V := '';
end;
SetLength(CtCustFieldTypeList, L);
CtCustFieldTypeList[L - 1] := S;
SetLength(CtCustFieldTypeDefList, L);
CtCustFieldTypeDefList[L - 1] := V;
end;
I := 0;
L := 0;
SetLength(CtCustDataTypeReplaces, L);
while True do
begin
Inc(I);
S := Trim(ini.ReadString('CustDataTypeReplaces', IntToStr(I), ''));
if S = '' then
Break;
Inc(L);
SetLength(CtCustDataTypeReplaces, L);
CtCustDataTypeReplaces[L - 1] := S;
end;
I := 0;
L := 0;
SetLength(CtTbNamePrefixDefs, L);
while True do
begin
Inc(I);
S := Trim(ini.ReadString('TbNamePrefixDefs', IntToStr(I), ''));
if S = '' then
Break;
Inc(L);
SetLength(CtTbNamePrefixDefs, L);
CtTbNamePrefixDefs[L - 1] := S;
end;
I := 0;
L := 0;
SetLength(CtCustFieldDataGenRules, L);
while True do
begin
Inc(I);
S := Trim(ini.ReadString('CustFieldDataGenRules', IntToStr(I), ''));
if S = '' then
Break;
Inc(L);
SetLength(CtCustFieldDataGenRules, L);
CtCustFieldDataGenRules[L - 1] := S;
end;
G_CheckForUpdates := ini.ReadBool('Options', 'CheckForUpdates', True);
FCurFileName := ini.ReadString('RecentFiles', 'CurFileName', '');
FAutoSaveMinutes := ini.ReadInteger('Options', 'AutoSaveMinutes', FAutoSaveMinutes);
FSaveTempFileOnExit := ini.ReadBool('Options', 'SaveTempFileOnExit',
FSaveTempFileOnExit);
FieldNameMaxDrawSize := ini.ReadInteger('Options', 'FieldNameMaxDrawSize',
FieldNameMaxDrawSize);
FieldTypeMaxDrawSize := ini.ReadInteger('Options', 'FieldTypeMaxDrawSize',
FieldTypeMaxDrawSize);
TableFieldMaxDrawCount := ini.ReadInteger('Options', 'TableFieldMaxDrawCount',
TableFieldMaxDrawCount);
G_MaxRowCountForTableData :=
ini.ReadInteger('Options', 'MaxRowCountForTableData', G_MaxRowCountForTableData);
G_CtSqlMaxFetchCount :=
ini.ReadInteger('Options', 'CtSqlMaxFetchCount', G_CtSqlMaxFetchCount);
G_HugeModeTableCount := ini.ReadInteger('Options', 'HugeModeTableCount',
G_HugeModeTableCount);
G_CreateSeqForOracle := ini.ReadBool('Options', 'CreateSeqForOracle',
G_CreateSeqForOracle);
G_BigIntForIntKeys := ini.ReadBool('Options', 'BigIntForIntKeys',
G_BigIntForIntKeys);
G_QuotReservedNames := ini.ReadBool('Options', 'QuotReservedNames',
G_QuotReservedNames);
G_QuotAllNames := ini.ReadBool('Options', 'QuotAllNames', G_QuotAllNames);
G_LogicNamesForTableData :=
ini.ReadBool('Options', 'LogicNamesForTableData', G_LogicNamesForTableData);
G_AddLimitToCursorSQL :=
ini.ReadBool('Options', 'AddLimitToCursorSQL', G_AddLimitToCursorSQL);
G_WriteConstraintToDescribeStr :=
ini.ReadBool('Options', 'WriteConstraintToDescribeStr', G_WriteConstraintToDescribeStr);
G_FieldGridShowLines := ini.ReadBool('Options', 'FieldGridShowLines',
G_FieldGridShowLines);
G_AddColCommentToCreateTbSql :=
ini.ReadBool('Options', 'AddColCommentToCreateTbSql', G_AddColCommentToCreateTbSql);
G_CreateIndexForForeignkey :=
ini.ReadBool('Options', 'CreateIndexForForeignkey', G_CreateIndexForForeignkey);
G_CreateForeignkeys :=
ini.ReadBool('Options', 'CreateForeignkeys', G_CreateForeignkeys);
G_HiveVersion :=
ini.ReadInteger('Options', 'HiveVersion', G_HiveVersion);
G_MysqlVersion :=
ini.ReadInteger('Options', 'MysqlVersion', G_MysqlVersion);
G_AutoCommit :=
ini.ReadBool('Options', 'AutoCommit', G_AutoCommit);
G_RetainAfterCommit :=
ini.ReadBool('Options', 'RetainAfterCommit', G_RetainAfterCommit);
G_ShowJdbcConsole :=
ini.ReadBool('Options', 'ShowJdbcConsole', G_ShowJdbcConsole);
G_EnableCustomPropUI := ini.ReadBool('Options', 'EnableCustomPropUI',
G_EnableCustomPropUI);
G_CustomPropUICaption := ini.ReadString('Options', 'CustomPropUICaption', '');
G_EnableAdvTbProp := ini.ReadBool('Options', 'EnableAdvTbProp',
G_EnableAdvTbProp);
G_EnableTbPropGenerate := ini.ReadBool('Options', 'EnableTbPropGenerate',
G_EnableTbPropGenerate);
G_EnableTbPropRelations := ini.ReadBool('Options', 'EnableTbPropRelations',
G_EnableTbPropRelations);
G_EnableTbPropData := ini.ReadBool('Options', 'EnableTbPropData',
G_EnableTbPropData);
G_EnableTbPropUIDesign := ini.ReadBool('Options', 'EnableTbPropUIDesign',
G_EnableTbPropUIDesign);
G_BackupBeforeAlterColumn :=
ini.ReadBool('Options', 'BackupBeforeAlterColumn', G_BackupBeforeAlterColumn);
G_TableDialogViewModeByDefault :=
ini.ReadBool('Options', 'TableDialogViewModeByDefault', G_TableDialogViewModeByDefault);
S := ini.ReadString('Options', 'OCILIB', '');
if S = '' then
S := ini.ReadString('Options', 'OCIDLL', '');
if S = '' then
begin
V := GetFolderPathOfAppExe();
V := FolderAddFileName(V, ocilib);
if FileExists(V) then
S := V;
end;
if S <> '' then
begin
OCILoadedLibrary := S;
//Windows.SetEnvironmentVariable('_NS_ORA_INSTANT_CLIENT', 'True');
//Windows.SetEnvironmentVariable('_NS_OCIDLL', PAnsiChar(S));
end;
S := ini.ReadString('Options', 'NLSLang', '');
G_OracleNlsLang := S;
if S <> '' then
begin
SetEnvVar('NLS_LANG', S);
end;
S := ini.ReadString('Options', 'MYSQLLIB', '');
if S = '' then
begin
V := GetFolderPathOfAppExe();
V := FolderAddFileName(V, mysqllib);
if FileExists(V) then
S := V;
end;
if S <> '' then
begin
MysqlLoadedLibrary := S;
end;
{$IFDEF USE_MSSQL}
S := ini.ReadString('Options', 'SQLSERVERLIB', '');
if S = '' then
begin
V := GetFolderPathOfAppExe();
V := FolderAddFileName(V, DBLIBDLL);
if FileExists(V) then
S := V;
end;
if S <> '' then
begin
DBLibLibraryName := S;
end;
{$ENDIF}
S := ini.ReadString('Options', 'POSTGRESLIB', '');
if S = '' then
begin
V := GetFolderPathOfAppExe();
V := FolderAddFileName(V, pqlib);
if FileExists(V) then
S := V;
end;
if S <> '' then
begin
Postgres3LoadedLibrary := S;
end;
S := ini.ReadString('Options', 'SQLITELIB', '');
if S = '' then
begin
V := GetFolderPathOfAppExe();
V := FolderAddFileName(V, Sqlite3Lib);
if FileExists(V) then
S := V;
end;
if S <> '' then
begin
SQLiteDefaultLibrary := S;
end;
G_OdbcCharset := ini.ReadString('Options', 'OdbcCharset', '');
G_LastMetaDbSchema := ini.ReadString('Options', 'LastMetaDbSchema', '');
//S := ini.ReadString('Options', 'LANG', '');
//if S<>'' then
// SetDefaultLang(S);
if ini.ReadBool('Options', 'EnableHttpServer', False) then
begin
Self.actHttpServer.Visible := True;
end;
FMainSplitterPos := ini.ReadInteger('MainForm', 'MainSplitterPos', FMainSplitterPos);
FStartMaximized := ini.ReadBool('MainForm', 'Maximized', False);
{$ifdef EZDML_CHATGPT}
G_ChatGPTKey := ini.ReadString('Options', 'ChatGPTKey', '');
if G_ChatGPTKey <> '' then
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actChatGPT.Tag := 2;
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actChatGPT.Visible := True;
end
else
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actChatGPT.Tag := 0;
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actChatGPT.Visible := True;//False;
end;
{$endif}
finally
ini.Free;
end;
end;
procedure TfrmMainDml.FormDestroy(Sender: TObject);
begin
try
FRecentFiles.Free;
FCustomTools.Free;
FCtDataModelList.Free;
if Assigned(FGlobalScriptor) then
FreeAndNil(FGlobalScriptor);
ClearCtMetaDbReg(True);
except
end;
end;
function TfrmMainDml.GetFastTmpFileName(fn: string): string;
var
dir: string;
begin
if IsTmpFile(fn) then
Result := ''
else
begin
dir := GetTmpDirForFile(fn);
fn := ExtractDmlFileName(fn);
fn := ChangeFileExt(fn, '') + '(0).~dmh0';
fn := FolderAddFileName(dir, fn);
Result := fn;
end;
end;
function TfrmMainDml.GetLastTmpFileName(fn: string): string;
var
dir: string;
begin
Result := '';
if IsTmpFile(fn) then
begin
Exit;
end;
dir := GetTmpDirForFile(fn);
if not DirectoryExists(dir) then
Exit;
fn := ExtractDmlFileName(fn);
fn := ChangeFileExt(fn, '.~dmh');
fn := FolderAddFileName(dir, fn);
fn := GetLastUsedFileName(fn);
Result := fn;
end;
function TfrmMainDml.GetNewTmpFileName(fn: string): string;
var
dir: string;
begin
if IsTmpFile(fn) then
Result := ''
else
begin
dir := GetTmpDirForFile(fn);
if not DirectoryExists(dir) then
ForceDirectories(dir);
fn := ExtractDmlFileName(fn);
fn := ChangeFileExt(fn, '.~dmh');
fn := FolderAddFileName(dir, fn);
fn := GetUnusedTmpFileName(fn);
Result := fn;
end;
end;
function TfrmMainDml.GetStatusPanelFileName(fn: string): string;
begin
if IsTmpFile(fn) then
Result := Format(srEzdmlTempFileFmt, [ExtractFileName(fn)])
else
Result := fn;
end;
function TfrmMainDml.GetTmpDirForFile(fn: string): string;
var
dir: string;
begin
if fn = '' then
Result := GetAppDefTempPath
else if IsTmpFile(fn) then
Result := ExtractFileDir(fn)
else
begin
dir := ExtractDmlFileDir(fn);
dir := StringReplace(dir, ':\', DirectorySeparator, []);
dir := StringReplace(dir, ':', DirectorySeparator, []);
Result := FolderAddFileName(GetAppDefTempPath(), dir);
end;
Result := TrimFileName(Result);
end;
function TfrmMainDml.ExtractDmlFileDir(fn: string): string;
var
ptr, eng, usr, db, doc, fid: String;
begin
if IsDbFile(fn) then
begin
if ParseDbFileName(fn, ptr, eng, usr, db, doc, fid) then
begin
Result := ptr;
if eng <>'' then
Result := Result + DirectorySeparator + eng;
if db <>'' then
Result := Result + DirectorySeparator + db;
if usr <>'' then
Result := Result + DirectorySeparator + usr;
if doc <>'' then
Result := Result + DirectorySeparator + doc;
if fid <>'' then
Result := Result + '.his' + DirectorySeparator + fid;
Result := ExtractFileDir(Result);
end
else
Result := ExtractFileDir(fn);
end
else
Result := ExtractFileDir(fn);
end;
function TfrmMainDml.ExtractDmlFileName(fn: string): string;
begin
if IsDbFile(fn) then
begin
if DirectorySeparator <> '/' then
fn := StringReplace(fn,'/', DirectorySeparator, [rfReplaceAll]);
end;
Result := ExtractFileName(fn);
end;
procedure TfrmMainDml.ImportFromFile(fn: string);
procedure ImportPDM;
begin
{$ifndef EZDML_LITE}
SetStatusBarMsg(Format(srEzdmlOpeningFileFmt, [GetStatusPanelFileName(fn)]));
Self.Refresh;
FCtDataModelList.Clear;
FFrameCtTableDef.Init(nil, True);
FCurFileName := '';
FProgressAll := 0;
FProgressCur := 0;
FWaitWnd := TfrmWaitWnd.Create(Self);
with TCtMetaPdmImporter.Create do
try
FWaitWnd.Init(srEzdmlOpenFile + ' ' + ExtractFileName(fn), srEzdmlOpening,
srEzdmlAbortOpening);
FWaitWnd.CheckAbort;
ModelList := FCtDataModelList;
FileName := fn;
DoImport;
Sleep(100);
FWaitWnd.CheckAbort;
Sleep(200);
FWaitWnd.CheckAbort;
finally
Free;
FWaitWnd.Release;
FWaitWnd := nil;
FFrameCtTableDef.Init(FCtDataModelList, False);
end;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
var
ext: string;
begin
CheckCanEditMeta;
ext := LowerCase(ExtractFileExt(fn));
if ext = '.pdm' then
begin
ImportPDM;
end;
SetStatusBarMsg(GetStatusPanelFileName(fn));
FCurFileName := '';
FAutoSaveCounter := 0;
CheckCaption;
end;
procedure TfrmMainDml.LoadFromDbFile(fn: string);
var
fs: TCtObjMemJsonSerialer;
begin
CheckCanEditMeta;
if FFileWorking then
Exit;
if Assigned(FWaitWnd) then
raise Exception.Create('wait wnd busy');
FFileWorking := True;
try
FCtDataModelList.Pack;
frmEzdmlDbFile.Caption := actLoadFromDb.Caption;
frmEzdmlDbFile.IsSaveMode:=False;
if not frmEzdmlDbFile.PrepareToLoadFile(fn) then
raise Exception.Create('DB file not ready: '+fn);
SetStatusBarMsg(Format(srEzdmlOpeningFileFmt, [GetStatusPanelFileName(fn)]));
Self.Refresh;
FProgressAll := 0;
FProgressCur := 0;
FWaitWnd := TfrmWaitWnd.Create(Self);
fs := TCtObjMemJsonSerialer.Create(True);
FFrameCtTableDef.IsInitLoading := False;
try
fs.RootName := 'DataModels';
fs.CurCtVer :=0;
FWaitWnd.Init(srEzdmlOpenFile + ' ' + frmEzdmlDbFile.ResultFileName, srEzdmlOpening,
srEzdmlAbortOpening);
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'DB_FILE_LOAD', frmEzdmlDbFile.ResultFileName, Self, nil);
end;
frmEzdmlDbFile.LoadFromDbFile(fs.Stream, frmEzdmlDbFile.ResultFileID);
fs.Stream.Seek(0, soFromBeginning);
try
FFrameCtTableDef.Init(nil, True);
except
end;
FFrameCtTableDef.IsInitLoading := True;
FCtDataModelList.LoadFromSerialer(fs);
FCurDmlFileName := '';
FFileDbConnectOk := True;
finally
fs.Free;
FWaitWnd.Release;
FWaitWnd := nil;
FFrameCtTableDef.IsInitLoading := False;
FFrameCtTableDef.Init(FCtDataModelList, False);
end;
if FCtDataModelList.ModelFileConfig.LastModel <> '' then
begin
TimerDelayCmd.Tag := 1;
TimerDelayCmd.Enabled := True;
end;
SetStatusBarMsg(GetStatusPanelFileName(fn));
FCurFileName := fn;
FCurDmlFileName := fn;
Self.RememberFileDateSize;
FAutoSaveCounter := 0;
CheckCaption;
finally
FFileWorking := False;
end;
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'DB_FILE_LOADED', FCurFileName, Self, nil);
end;
end;
function TfrmMainDml.CheckDbFileState(fn: string; bForce: Boolean): Integer;
var
fileSize: Integer;
fileDate: TDateTime;
begin
//检查数据库文件状态
//返回:0未连接 1连接失败 2不存在 3存在
Result := frmEzdmlDbFile.CheckDbFileState(fn, fileSize, fileDate, bForce);
if Result <= 1 then
FFileDbConnectOk := False;
end;
function TfrmMainDml.IsTmpFile(fn: string): boolean;
var
ext: string;
begin
Result := False;
ext := ExtractFileExt(fn);
ext := LowerCase(ext);
if (ext = '.~dmh') or (ext = '.~dmh0') or (ext = '.~dmj') then
Result := True;
if IsDbFile(fn) then
begin
if IsDbHistFile(fn) then //历史文件?
begin
Result := True;
Exit;
end;
Result := False;
end;
end;
function TfrmMainDml.IsDbFile(fn: string): boolean;
begin
Result := False;
if Pos('db://', fn)=1 then
Result := True;
end;
function TfrmMainDml.IsDbHistFile(fn: string): boolean;
var
ptr, eng, usr, db, doc, fid: string;
begin
Result := False;
if not IsDbFile(fn) then
Exit;
if ParseDbFileName(fn, ptr, eng, usr, db, doc, fid) then
if fid <> '' then //历史文件?
Result := True
end;
function CompareStream(// 比较两个流是否相等
mStream1, mStream2: TStream // 两个流
): boolean; // 返回两个流是否相等
var
vBuffer1, vBuffer2: array[0..$1000 - 1] of char;
vLength1, vLength2: integer;
begin
Result := mStream1 = mStream2;
if Result then
Exit;
if not Assigned(mStream1) or not Assigned(mStream2) then
Exit; // 其中一个为空
while True do
begin
vLength1 := mStream1.Read(vBuffer1, SizeOf(vBuffer1));
vLength2 := mStream2.Read(vBuffer2, SizeOf(vBuffer2));
if vLength1 <> vLength2 then
Exit;
if vLength1 = 0 then
Break;
if not CompareMem(@vBuffer1[0], @vBuffer2[0], vLength1) then
Exit;
end;
Result := True;
end;
function IsSameFileContent(// 比较两个文件是否相等
mFileName1, mFileName2: string // 两个文件
): boolean; // 返回两个文件是否相等
var
vFileStream1, vFileStream2: TFileStream;
fn1, fn2: string;
begin
Result := False;
if not FileExists(mFileName1) or not FileExists(mFileName2) then
Exit; // 其中一个文件不存在
fn1 := ExpandFileName(mFileName1);
fn2 := ExpandFileName(mFileName2);
if LowerCase(fn1) = LowerCase(fn2) then// 两个文件名是否相同
begin
Result := True;
Exit;
end;
if GetDocFileSize(mFileName1) <> GetDocFileSize(mFileName2) then
// 文件大小是否一致
Exit;
vFileStream1 := TFileStream.Create(mFileName1, fmOpenRead or fmShareDenyNone);
vFileStream2 := TFileStream.Create(mFileName2, fmOpenRead or fmShareDenyNone);
try
Result := CompareStream(vFileStream1, vFileStream2);
// 比较两个文件内容是否相同
finally
vFileStream1.Free;
vFileStream2.Free;
end;
end;
procedure TfrmMainDml.SaveDMLFastTmpFile(bForceSaveHuge: Boolean);
var
fn, fastFn: string;
vFileDate: TDateTime;
begin
if FCtDataModelList.TableCount <= 0 then
Exit;
if not bForceSaveHuge then
if FCtDataModelList.IsHuge then
Exit;
if not FSaveTempFileOnExit then
begin
Exit;
end;
try
fn := SaveDMLToTmpFile(); //这一次保存的临时文件
if fn <> '' then
begin
fastFn := GetFastTmpFileName(FCurFileName); //上一次保存的临时文件
if (fastFn <> '') and not FileExists(fastFn) then
begin
//快速加载文件不存在,直接复制一份
CopyFile(PChar(fn), PChar(fastfn), False);
fn := fastfn;
end
else if (fastFn <> '') and (fastFn <> fn) then
//将新文件覆盖零号文件(启动时会自动加载)
begin
if IsSameFileContent(fastFn, fn) then
begin
fn := fastFn;
end
else
begin
DeleteFile(fastFn);
CopyFile(PChar(fn), PChar(fastfn), False);
fn := fastfn;
end;
end;
if fn <> '' then
begin
//数据库文件,如果数据库断开取不到文件时间,直接设置上一次的时间并退出
if IsDbFile(FCurFileName) then
begin
if CheckDbFileState(FCurFileName, False) <= 2 then //数据库断开?
begin
SetFileAges(fn, FCurFileDate);
Exit;
end;
if not GetDmlFileDate(FCurFileName, vFileDate) then //取不到数据库文件日期?
begin
SetFileAges(fn, FCurFileDate);
Exit;
end;
end;
if GetDmlFileDate(FCurFileName, vFileDate) then
SetFileAges(fn, vFileDate);
end;
end;
except
end;
end;
function TfrmMainDml.SaveDMLToTmpFile: string;
var
lastFn, fn, sfn, sts: string;
begin
Result := '';
if FCtDataModelList.TableCount <= 0 then
Exit;
sfn := FCurFileName;
fn := sfn;
if fn = '' then
fn := GetConfFileOfApp('.dmh');
if IsTmpFile(fn) then
Exit;
lastFn := GetLastTmpFileName(fn); //上一次保存的临时文件
if not IsTmpFile(fn) then
fn := GetNewTmpFileName(fn);
sts := StatusBar1.SimpleText;
FIsAutoSaving := True;
try
SaveToFile(fn);
finally
FIsAutoSaving := False;
end;
FCurFileName := sfn;
CheckCaption;
Result := fn;
if not FileExists(lastFn) then
begin
lastFn := '';
//快速加载文件不存在
end
//判断两次的文件是否相同
else if (lastFn <> '') and (lastFn <> fn) and IsSameFileContent(lastFn, fn) then
begin
//完全相同的话,说明本次备份是多余,直接删除之
DeleteFile(Result);
Result := lastFn;
SetStatusBarMsg(sts);
end;
end;
procedure TfrmMainDml.SaveIni;
var
ini: TIniFile;
begin
ini := TIniFile.Create(GetConfFileOfApp);
try
if FCurFileName = '' then
ini.WriteString('RecentFiles', 'CurFileName', FCurFileName);
FMainSplitterPos := Self.FFrameCtTableDef.PanelCttbTree.Width;
if FMainSplitterPos > 20 then
ini.WriteInteger('MainForm', 'MainSplitterPos', FMainSplitterPos);
if Self.WindowState = wsMaximized then
ini.WriteBool('MainForm', 'Maximized', True)
else
ini.WriteBool('MainForm', 'Maximized', False);
ini.WriteString('Options', 'LastMetaDbSchema', G_LastMetaDbSchema);
finally
ini.Free;
end;
end;
procedure TfrmMainDml.SaveToFile(fn: string);
var
fs: TCtObjSerialer;
//I: Integer;
begin
if FFileWorking then
Exit;
FFileWorking := True;
try
if Assigned(FWaitWnd) then
raise Exception.Create('wait wnd busy');
try
if not FIsAutoSaving then
if not FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameCtTableList.TreeViewCttbs.CanFocus then
FFrameCtTableDef.FFrameCtTableList.TreeViewCttbs.SetFocus;
except
end;
SetStatusBarMsg(Format(srEzdmlSaveingFileFmt, [GetStatusPanelFileName(fn)]));
Self.Refresh;
if FileExists(fn) then
DeleteFile(fn);
fs := EzdmlCreateCtObjSerialer(fn, True);
try
fs.RootName := 'DataModels';
FCtDataModelList.Pack;
FProgressAll := 0;
FProgressCur := 0;
if FIsAutoSaving and not FCtDataModelList.IsHuge then
//自动保存:超过1万个字段才显示进度
begin
FWaitWnd := nil;
Screen.Cursor := crAppStart;
end
else
FWaitWnd := TfrmWaitWnd.Create(Self);
try
if Assigned(FWaitWnd) then
FWaitWnd.Init(srEzdmlSaveFile + ' ' + ExtractFileName(fn), srEzdmlSaving,
srEzdmlAbortSaving);
{for I := 0 to FTableList.Count - 1 do
FTableList[I].MetaFields.Pack;}
FCtDataModelList.SaveToSerialer(fs);
finally
if Assigned(FWaitWnd) then
FWaitWnd.Release;
FWaitWnd := nil;
Screen.Cursor := crDefault;
end;
finally
fs.Free;
end;
SetStatusBarMsg(srEzdmlSaved + GetStatusPanelFileName(fn) + ' ' + TimeToStr(Now));
FCurFileName := fn;
RememberFileDateSize;
FAutoSaveCounter := 0;
if not FIsAutoSaving then
CheckCaption;
finally
FFileWorking := False;
end;
end;
procedure TfrmMainDml.SetRecentFile(fn: string);
var
I: integer;
S: string;
ini: TIniFile;
begin
if fn = '' then
Exit;
if IsTmpFile(fn) then
Exit;
ini := TIniFile.Create(GetConfFileOfApp);
try
I := 0;
FRecentFiles.Clear;
while True do
begin
Inc(I);
S := ini.ReadString('RecentFiles', IntToStr(I), '');
if S = '' then
Break;
FRecentFiles.Add(S);
end;
S := LowerCase(fn);
for I := 0 to FRecentFiles.Count - 1 do
if LowerCase(FRecentFiles[I]) = S then
begin
FRecentFiles.Delete(I);
Break;
end;
FRecentFiles.Insert(0, fn);
ini.EraseSection('RecentFiles');
for I := 0 to FRecentFiles.Count - 1 do
ini.WriteString('RecentFiles', IntToStr(I + 1), FRecentFiles[I]);
ini.WriteString('RecentFiles', 'CurFileName', FCurFileName);
finally
ini.Free;
end;
RecreateRecentMn;
end;
procedure TfrmMainDml.SetStatusBarMsg(msg: string; tp: integer);
begin
if (tp < 0) or (tp >= StatusBar1.Panels.Count) then
tp := 0;
if (tp=0) and (msg='') then
msg := ExtractDmlFileName(Self.FCurFileName);
if StatusBar1.Panels[tp].Text <> msg then
begin
StatusBar1.Panels[tp].Text := msg;
StatusBar1.Refresh;
end;
end;
procedure TfrmMainDml.TimerAutoSaveTimer(Sender: TObject);
begin
if csDestroying in Self.ComponentState then
Exit;
if FIsAutoSaving then
Exit;
if FCheckingFileDate then
Exit;
if Application.ModalLevel > 0 then
Exit;
if IsDbFile(FCurFileName) and not IsDbHistFile(FCurFileName) then
_OnAppActivate(nil);
if (FAutoSaveMinutes = 0) then
Exit;
//hw := GetForegroundWindow;
// if GetWindowThreadProcessId(hw, nil) <> MainThreadID then
// Exit;
Inc(FAutoSaveCounter);
if (FAutoSaveCounter / 6) < FAutoSaveMinutes then
Exit;
if Assigned(FWaitWnd) then
Exit;
FAutoSaveCounter := 0;
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'FILE_AUTOSAVE', FCurDmlFileName, Self, nil);
end;
if not FCtDataModelList.IsHuge then
SaveDmlToTmpFile;
end;
procedure TfrmMainDml.TimerInitTimer(Sender: TObject);
var
fn, ext: string;
tp: Integer;
begin
TimerInit.Enabled := False;
if csDestroying in Self.ComponentState then
Exit;
if Application.Terminated then
Exit;
if (Top + Height + ScaleDPISize(80)) > Screen.DesktopHeight then
begin
tp := Screen.DesktopHeight - Height - ScaleDPISize(80);
if tp >= 0 then
Top := tp
else
begin
Top := 0;
end;
end;
if FStartMaximized then
Self.WindowState:=wsMaximized;
CheckShowNewVersionInfo(False);
if FMainSplitterPos >= 20 then
Self.FFrameCtTableDef.PanelCttbTree.Width := FMainSplitterPos;
if ParamStr(1) <> '' then
begin
ext := ExtractFileExt(ParamStr(1));
if LowerCase(ext) <> '.pas' then
begin
PromptOpenFile(ParamStr(1));
if ParamStr(2) <> '' then
begin
if GetDmlScriptType(ParamStr(2)) <> '' then
ExecDmlScript(ParamStr(2));
end;
end;
end
else if FCurFileName <> '' then
begin
fn := FCurFileName;
FCurFileName := '';
try
TryLockFile(fn, False);
except
Exit;
end;
if IsDbFile(fn) and not IsDbHistFile(fn) then
begin
if CheckDbFileState(fn, True) <= 2 then
begin
if IsCtDbConnected then
actLoadFromDb.Execute;
Exit;
end;
end;
if TryLoadFromTmpFile(fn) then
begin
end
else if FRecentFiles.Count > 0 then
actOpenLastFile1.Execute;
end
else if FRecentFiles.Count > 0 then
begin
//fn := FRecentFiles[0];
//if Application.MessageBox(PChar(Format(srEzdmlOpenLastFileFmt, [fn])),
// PChar(srEzdmlNew), MB_OKCANCEL or MB_ICONQUESTION) = IDOK then
{ actOpenLastFile1.Execute;
if FCurFileName <> '' then
Self.SaveIni; }
end
else
begin
fn := GetFolderPathOfAppExe;
fn := FolderAddFileName(fn, 'demo.dmj');
fn := GetConfigFile_OfLang(fn);
if FileExists(fn) then
begin
if Application.MessageBox(PChar(srEzdmlPromptOpenDemoFile),
PChar(Application.Title), MB_OKCANCEL or MB_ICONQUESTION) = idOk then
begin
PromptOpenFile(fn);
end;
end
else
begin
//fn :='/Users/admin/Documents/ezdml_x/demo.dmj';
// PromptOpenFile(fn);
end;
end;
if FCurFileName = '' then
FFrameCtTableDef.Init(FCtDataModelList, False);
end;
function TfrmMainDml.TryLoadFromTmpFile(sfn: string): boolean;
var
fn, S, usr, memo: string;
bCheck: boolean;
iBtns: Integer;
vFileDate1, vFileDate2: TDateTime;
begin
Result := False;
if not FSaveTempFileOnExit then
Exit;
if not IsDBFile(sfn) then
if not FileExists(sfn) then
Exit;
fn := GetFastTmpFileName(sfn);
if fn = '' then
Exit;
if not FileExists(fn) then
Exit;
bCheck := False;
if GetDmlFileDate(sfn, vFileDate1) and
GetDmlFileDate(fn, vFileDate2) then
begin
if Abs(vFileDate1 - vFileDate2) > 2 / 24 / 60 / 60 then
bCheck := True;
end
else
bCheck := True;
if bCheck then
begin
iBtns := MB_OK;
S := Format(srEzdmlTmpFileIgnoredFmt, [fn]);
if IsDbFile(sfn) then
begin
usr:='';
memo:='';
if frmEzdmlDbFile <> nil then
if frmEzdmlDbFile.GetDbFileModifierInfo(sfn, usr, memo) then
begin
S := Format(srEzdmlDbTmpFileIgnoredFmt, [fn, usr, memo]);
iBtns := MB_OKCANCEL;
end;
end;
case (Application.MessageBox(PChar(S),
PChar(srEzdmlOpenFile), iBtns or MB_ICONWARNING)) of
idOk: fn := sfn;
else
Abort;
end;
end;
if IsDBFile(fn) then
LoadFromDbFile(fn)
else
LoadFromFile(fn);
if IsDBFile(sfn) then
FFileDbConnectOk := True;
FCurFileName := sfn;
FCurDmlFileName := FCurFileName;
RememberFileDateSize;
CheckCaption;
Result := True;
end;
procedure CallScriptFunctionAsMethod;
var
S: string;
begin
with TPSScript.Create(nil) do
try
Script.Clear;
Script.Add(
'function Test(s,par1,par2,par3,par4:string): string; begin Result := ''Test Results:2''+s+par1+par2+par3+par4;end; begin end.');
if not Compile() then
ShowMessage('err1');
S := ExecuteFunction(['INDATA', 'ss', 'p222', 'p333', 'p444'], 'Test');
ShowMessage(S);
finally
Free;
end;
end;
procedure TfrmMainDml.CheckReloadGlobalScript;
var
FileTxt: TStrings;
fn, S: string;
bSuccess: boolean;
ce: TPSScript;
begin
//CallScriptFunctionAsMethod;
//Exit;
GProc_OnEzdmlGenTbSqlEvent := nil;
GProc_OnEzdmlGenDbSqlEvent := nil;
GProc_OnEzdmlGenFieldTypeDescEvent := nil;
GProc_OnEzdmlGenAlterFieldEvent := nil;
GProc_OnEzdmlGenDataSqlEvent := nil;
GProc_OnEzdmlCmdEvent := nil;
if Assigned(FGlobalScriptor) then
FreeAndNil(FGlobalScriptor);
{$ifndef EZDML_LITE}
fn := DEF_GSCRIPT_FN;
S := GetFolderPathOfAppExe;
S := FolderAddFileName(S, fn);
if not FileExists(S) then
Exit;
FGlobalScriptor := TDmlGlobalPasScript.Create;
TDmlPasScriptor(FGlobalScriptor).ActiveFile := S;
FileTxt := TStringList.Create;
try
FileTxt.LoadFromFile(S);
S := FileTxt.Text;
finally
FileTxt.Free;
end;
bSuccess := False;
with TDmlGlobalPasScript(FGlobalScriptor) do
try
ce := GetPS;
ce.UsePreProcessor := True;
if not Compile('PASCAL_SCRIPT', S) then
raise Exception.Create(DEF_GSCRIPT_FN + ' compile failed!'#13#10 + StdOutPut.Text);
Exec('PASCAL_SCRIPT', S);
//GProc_OnEzdmlGenTbSqlEvent := TOnEzdmlGenTbSqlEvent(ce.GetProcMethod('OnEzdmlGenTbSqlEvent'));
//GProc_OnEzdmlGenDbSqlEvent := TOnEzdmlGenDbSqlEvent(ce.GetProcMethod('OnEzdmlGenDbSqlEvent'));
//GProc_OnEzdmlGenFieldTypeDescEvent := TOnEzdmlGenFieldTypeDescEvent(ce.GetProcMethod('OnEzdmlGenFieldTypeDescEvent'));
//GProc_OnEzdmlGenAlterFieldEvent := TOnEzdmlGenAlterFieldEvent(ce.GetProcMethod('OnEzdmlGenAlterFieldEvent'));
//GProc_OnEzdmlCmdEvent := TOnEzdmlCmdEvent(ce.GetProcMethod('OnEzdmlCmdEvent'));
TakeGlobalEvents;
//S:=GProc_OnEzdmlCmdEvent( 'TEST222222222222222', 'pp111','p222',Self,nil);
//ShowMessage(S);
bSuccess := True;
finally
if not bSuccess then
FreeAndNil(FGlobalScriptor);
end;
{$else}
FGlobalScriptor := TDmlGlobalPasScriptLite.Create;
TDmlGlobalPasScriptLite(FGlobalScriptor).TakeGlobalEvents;
{$endif}
end;
procedure TfrmMainDml.CheckForUpdates(bForceNow: boolean);
var
ini: TIniFile;
dt: TDateTime;
uid, url, pf, jstr, opt, ver: string;
cc: integer;
jo: TJSONObject;
begin
uid := GetMyComputerId;
ini := TIniFile.Create(GetConfFileOfApp);
try
cc := ini.ReadInteger('Updates', 'tk', 0);
Inc(cc);
ini.WriteInteger('Updates', 'tk', cc);
if not bForceNow then
begin
if not G_CheckForUpdates then
Exit;
dt := ini.ReadDateTime('Updates', 'LastCheckOkDate', 0);
if (Now - dt) < 8 then
Exit;
dt := ini.ReadDateTime('Updates', 'LastCheckDate', 0);
if (Now - dt) < 1 then
Exit;
end;
ini.WriteDateTime('Updates', 'LastCheckDate', Now);
{$IFDEF Windows}
{$ifdef WIN32}
pf := 'win32';
{$else}
pf := 'win64';
{$endif}
{$ELSE}
{$IFDEF Darwin}
pf := 'mac64';
{$ELSE}
pf := 'linux64';
{$ENDIF}
{$ENDIF}
url := 'http://www.ezdml.com/up.html?app=ezdml&platform=' + pf +
'&ver=' + srEzdmlVersionNum + '&uid=' + uid + '&tk=' + IntToStr(cc);
//Toast(srEzdmlCheckingForUpdates+#10+url, 1000);
{
"app": "ezdml",
"platform": "win32",
"ver": "3.09",
"date": "2021-04-11",
"desc": "2021-04-11 V3.09: new version, bugs fixed",
"detail_url": "http://www.ezdml.com/"
}
if bForceNow then
begin
opt := '[SHOW_PROGRESS]';
opt := opt + '[WAIT_TICKS=0][READ_TIMEOUT=20000]';
opt := opt + '[MSG=' + srEzdmlCheckingForUpdates + ']';
end
else
opt := '[READ_TIMEOUT=1800]';
try
try
if not bForceNow then
begin
Self.Hide;
Application.ProcessMessages;
end;
except
end;
jstr := GetUrlData_Net(url, '', opt);
if jstr = '' then
Exit;
{lres := jstr;
lres := StringReplace(lres, #13 ,' ', [rfReplaceAll]);
lres := StringReplace(lres, #10 ,' ', [rfReplaceAll]);
ini.WriteString('Updates', 'LastResult', lres); }
jo := TJSONObject.Create(jstr);
ver := jo.optString('ver');
ini.WriteString('Updates', 'NewVerNum', ver);
ini.WriteString('Updates', 'NewVerDate', jo.optString('date'));
ini.WriteString('Updates', 'NewVerDesc', jo.optString('desc'));
ini.WriteString('Updates', 'NewVerUrl', jo.optString('detail_url'));
jo.Free;
ini.WriteDateTime('Updates', 'LastCheckOkDate', Now);
except
on E: Exception do
//lres := 'Error: '+E.Message;
end;
//showmessage(lres);
finally
ini.Free;
end;
end;
procedure TfrmMainDml.CheckShowNewVersionInfo(bForceNow: boolean);
var
sCurVer, sNewVer, sLastPromptVer, sDesc: string;
fCurVer, fNewVer: Double;
ini: TIniFile;
begin
if not bForceNow and not G_CheckForUpdates then
Exit;
ini := TIniFile.Create(GetConfFileOfApp);
try
sCurVer := srEzdmlVersionNum;
fCurVer := StrToFloat(sCurVer);
sNewVer := ini.ReadString('Updates', 'NewVerNum', '');
fNewVer := StrToFloatDef(sNewVer, fCurVer);
if (sNewVer = '') or (sNewVer = sCurVer) or (fNewVer < fCurVer) then
begin
if not bForceNow then
Exit;
lbNewVerInfo.Caption := srEzdmlNoUpdateFound;
lbNewVerInfo.Tag := 2;
lbNewVerInfo.Hint := '';
end
else
begin
sLastPromptVer := ini.ReadString('Updates', 'LastPromptVer', '');
if not bForceNow then
begin
if (sNewVer = sLastPromptVer) then
Exit;
if StrToFloatDef(sCurVer, 0) >= StrToFloatDef(sNewVer, 0) then
Exit;
end;
sDesc := ini.ReadString('Updates', 'NewVerDesc', '');
if sDesc = '' then
sDesc := ini.ReadString('Updates', 'NewVerDate', '') + ' ' + sNewVer + ' released.';
lbNewVerInfo.Caption := sDesc;
lbNewVerInfo.Tag := 1;
lbNewVerInfo.Hint := ini.ReadString('Updates', 'NewVerUrl', '');
end;
PanelNewVerHint.Left := 4;
PanelNewVerHint.Top := Self.StatusBar1.Top - PanelNewVerHint.Height - 4;
PanelNewVerHint.Show;
PanelNewVerHint.BringToFront;
finally
ini.Free;
end;
end;
function TfrmMainDml.GetDmlFileDate(fn: string; var vFileDate: TDateTime
): boolean;
var
vFileSize: Integer;
begin
Result := GetDmlFileDateAndSize(fn, vFileSize, vFileDate);
end;
function TfrmMainDml.GetDmlFileDateAndSize(fn: string; var vFileSize: Integer;
var vFileDate: TDateTime): boolean;
var
age: longint;
begin
Result := False;
vFileSize := 0;
vfileDate := Now;
if IsDbFile(fn) then
begin
if IsDbHistFile(fn) then
Exit;
if frmEzdmlDbFile.CheckDbFileState(fn, vfileSize, vfileDate, True) > 2 then
begin
Result := True;
end;
Exit;
end;
age := FileAge(fn);
if age = -1 then
Exit;
vFileDate := FileDateToDateTime(age);
vFileSize := GetDocFileSize(fn);
Result := True;
end;
function TfrmMainDml.IsShortcut(var Message: TLMKey): boolean;
begin
Result := inherited IsShortcut(Message);
if not Result then
if Assigned(FFrameCtTableDef) and Assigned(FFrameCtTableDef.FFrameDMLGraph)
and FFrameCtTableDef.PanelDMLGraph.Visible then
Result := FFrameCtTableDef.FFrameDMLGraph.IsShortcut(Message);
end;
procedure TfrmMainDml.TryLockFile(fn: string; bAsk: boolean);
begin
if FFileLockMutex <> nil then
begin
FreeAndNil(FFileLockMutex);
end;
if fn = '' then
Exit;
FFileLockMutex := TCtMutex.Create(fn);
if not FFileLockMutex.Acquire(400) then
begin
{$ifdef WINDOWS}
bAsk := False;
{$else}
{$endif}
if not bAsk then
begin
FreeAndNil(FFileLockMutex);
raise Exception.Create(Format(srEzdmlFileAlreadyOpenedFmt, [fn]));
end
else if Application.MessageBox(
PChar(Format(srEzdmlConfirmAlreadyOpenedFileFmt, [fn])),
PChar(Application.Title), MB_YESNOCANCEL or MB_DEFBUTTON2 or MB_ICONWARNING) <>
idYes then
begin
FreeAndNil(FFileLockMutex);
Abort;
end
else
FFileLockMutex.Acquire(40, True);
end;
end;
procedure TfrmMainDml._OnAppActivate(Sender: TObject);
procedure CloseSubForms;
var
I, L: Integer;
frm: TForm;
cFrms: array of TForm;
begin
L := 0;
SetLength(cFrms, L);
for I:= Screen.FormCount - 1 downto 0 do
begin
frm := Screen.Forms[I];
if frm <> Self then
if (frm.Parent = nil) and frm.Visible and frm.Showing then
begin
L := L+1;
SetLength(cFrms, L);
cFrms[L-1] := frm;
end;
end;
EditMetaForceRelease;
for I:=0 to L - 1 do
begin
frm := cFrms[I];
if frm is TfrmCtTableProp then
TfrmCtTableProp(frm).ForceRelease
else
frm.Close;
end;
end;
var
S, usr, memo: String;
begin
if FFileWorking then
Exit;
if Application.ModalLevel > 0 then
begin
Self.Tag := 55678;
Exit;
end;
Self.Tag := 0;
if FCheckingFileDate then
Exit;
FCheckingFileDate := True;
try
try
if not CheckCurFileDateSizeChanged then
Exit;
except
Exit;
end;
if IsDbFile(FCurFileName) and not IsDbHistFile(FCurFileName) then
begin
if frmEzdmlDbFile = nil then
Exit;
if not frmEzdmlDbFile.GetDbFileModifierInfo(FCurFileName, usr, memo) then
Exit;
S := Format(srEzdmlPromptReloadDbFileChanged, [usr, memo]);
//if Application.MessageBox(PChar(S),
// PChar(Application.Title), MB_OKCANCEL or MB_ICONWARNING) <> idOk then
//begin
// Self.RememberFileDateSize;
// Exit;
//end;
if ShowMessageOnTop(S, Application.Title) <> mrOk then
begin
Self.RememberFileDateSize;
Exit;
end;
end
else
begin
//if Application.MessageBox(PChar(srEzdmlPromptReloadOnFileDateSizeChanged),
// PChar(Application.Title), MB_OKCANCEL or MB_ICONWARNING) <> idOk then
//begin
// Self.RememberFileDateSize;
// Exit;
//end;
if ShowMessageOnTop(srEzdmlPromptReloadOnFileDateSizeChanged, Application.Title) <> mrOk then
begin
Self.RememberFileDateSize;
Exit;
end;
end;
SaveDmlToTmpFile;
CloseSubForms;
FCtDataModelList.Clear;
FFrameCtTableDef.Init(FCtDataModelList, True);
PromptOpenFile(FCurFileName, True);
finally
FCheckingFileDate := False;
end;
end;
procedure TfrmMainDml._OnDbFileMemoChanged(Sender: TObject; fn: string);
begin
if ExtractDmlFileName(fn)=ExtractDmlFileName(FCurFileName) then
RememberFileDateSize;
end;
procedure TfrmMainDml._OnCustomToolsClick(Sender: TObject);
function GetCustomToolsDir: string;
begin
Result := GetFolderPathOfAppExe('CustomTools');
end;
var
fn: string;
{$ifdef EZDML_LITE}
ScLt : TDmlPasScriptorLite;
AOutput: TStrings;
{$endif}
begin
if Sender is TMenuItem then
begin
fn := TMenuItem(Sender).Hint;
{$ifdef EZDML_LITE}
ScLt := CreatePsLiteScriptor(fn, 'Tool');
if ScLt <> nil then
begin
AOutput := TStringList.Create;
try
with ScLt do
begin
Init('DML_SCRIPT', FFrameCtTableDef.GetCurTable, AOutput, nil);
Exec('DML_SCRIPT', '');
end;
finally
AOutput.Free;
ScLt.Free;
end;
end;
Exit;
{$endif}
//ext := ExtractFileExt(fn);
fn := FolderAddFileName(GetCustomToolsDir, fn);
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MENU_ACTION', 'Tools_CustomMenu', fn, Sender, nil);
end;
if GetDmlScriptType(fn) <> '' then
ExecDmlScript(fn)
else
CtOpenDoc(PChar(fn)); { *Converted from ShellExecute* }
end;
end;
procedure TfrmMainDml.CallAI(Act: Integer);
begin
EzdmlMenuActExecuteEvt('Model_ChatGPT');
{$ifdef EZDML_CHATGPT}
if Act>1 then
if not FFrameCtTableDef.PanelDMLGraph.Visible then
raise Exception.Create(srDmlGptGenShowGraphTip);
FFrameCtTableDef.FFrameDMLGraph.CheckSelectedTb;
if Act>=3 then
if FFrameCtTableDef.FFrameDMLGraph.GetSelectedTable=nil then
raise Exception.Create(srDmlGptGenSelectTip);
CheckCanEditMeta;
FCtDataModelList.Pack;
if not FCtDataModelList.IsHuge then
SaveDmlToTmpFile;
FCtDataModelList.ModelFileConfig.LastModel := '';
if ShowChatGPTForm(Act) then
begin
if Act=1 then
begin
FFrameCtTableDef.FFrameCtTableList.actRefresh.Execute;
if FCtDataModelList.ModelFileConfig.LastModel <> '' then
begin
TimerDelayCmd.Tag := 11;
TimerDelayCmd.Enabled := True;
end;
end
else if Act=2 then
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actRefresh.Execute;
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actBestFit.Execute;
end
else
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actRefresh.Execute;
end;
end;
{$endif}
end;
procedure TfrmMainDml._OnDMLObjProgress(Sender: TObject;
const Prompt: string; Cur, All: integer; var bContinue: boolean);
begin
if Assigned(FWaitWnd) then
begin
if (Prompt = '') and (Cur = 0) then
if FProgressAll = 0 then
begin
FProgressCur := 0;
FProgressAll := All;
end;
if FProgressAll > 0 then
begin
if Sender is TCtMetaTableList then
Inc(FProgressCur);
FWaitWnd.SetPercentMsg(FProgressCur * 100 / FProgressAll, Prompt, True);
end
else
begin
if All > 0 then
FWaitWnd.SetPercentMsg(Cur * 100 / All, Prompt, True)
else
FWaitWnd.CheckCanceled;
end;
if FWaitWnd.Canceled then
bContinue := False;
end;
end;
procedure TfrmMainDml._OnRecentFileClick(Sender: TObject);
var
fn: string;
begin
if Sender is TMenuItem then
begin
fn := TMenuItem(Sender).Hint;
if FCurFileName = fn then
Exit;
PromptOpenFile(fn);
end;
end;
procedure TfrmMainDml._WMZ_CUSTCMD(var msg: TMessage);
var
tb: TCtMetaTable;
begin
if msg.wParam = 1 then //打开文件
begin
PromptOpenFile(FGlobeOpeningFile);
FGlobeOpeningFile := '';
Exit;
end;
if msg.wParam = 2 then //切换模型图/表属性
begin
actToggleTableView.Execute;
Exit;
end;
if msg.wParam = 3 then //生成代码
begin
if msg.LParam = 1 then
actGenerateDatabase.Execute
else if msg.LParam = 2 then
actGenerateCode.Execute
else if msg.LParam = 3 then
actGenerateTestData.Execute
else if msg.LParam = 4 then
actSqlTool.Execute;
Exit;
end;
if msg.wParam = 4 then //跳到下一个模型图
begin
if msg.lParam = 1 then
FFrameCtTableDef.FFrameCtTableList.FocusSibling(False)
else if msg.lParam = 2 then
FFrameCtTableDef.FFrameCtTableList.FocusSibling(True);
Exit;
end;
if msg.wParam = 5 then //从查看表属性转到修改
begin
Application.ProcessMessages;
Application.Idle(True);
if Assigned(Proc_ShowCtTableProp) then
if G_WMZ_CUSTCMD_Object <> nil then
begin
tb := TCtMetaTable(G_WMZ_CUSTCMD_Object);
if Proc_ShowCtTableProp(tb, msg.lParam = 1, False) then
begin
if FFrameCtTableDef.PanelDMLGraph.Showing then
begin
FFrameCtTableDef.FFrameDMLGraph.ReloadTbInfo(tb);
end;
FFrameCtTableDef._OnCtTablePropChange(2, tb, nil, '');
end;
end;
Exit;
end;
if msg.wParam = 6 then //修改表属性事件:检查是否需要刷新
begin
tb := TCtMetaTable(G_WMZ_CUSTCMD_Object);
G_WMZ_CUSTCMD_Object := nil;
if (tb<>nil) and (FFrameCtTableDef.GetCurTable = tb) then
FFrameCtTableDef.RefreshProp;
Exit;
end;
if msg.wParam = 7 then //显示设置
begin
actSettings.Execute;
Exit;
end;
if msg.wParam = 8 then //ChatGPT
begin
CallAI(msg.LParam);
Exit;
end;
if msg.wParam = 9 then //检查文件变更
begin
Self._OnAppActivate(nil);
Exit;
end;
if msg.wParam = 10 then //设置
begin
if ShowEzdmlSettings(msg.LParam) then
begin
LoadIni;
end;
Exit;
end;
if msg.wParam = 11 then //分享
begin
if msg.lParam = 1 then
actShareFile.Execute
else if msg.lParam = 2 then
actOpenUrl.Execute
else if msg.lParam = 101 then
actOpenFile.Execute
else if msg.lParam = 102 then
actLoadFromDb.Execute
else if msg.lParam = 103 then
actOpenUrl.Execute
else if msg.lParam = 201 then
actSaveFileAs.Execute
else if msg.lParam = 202 then
actSaveToDb.Execute
else if msg.lParam = 203 then
actShareFile.Execute;
Exit;
end;
end;
procedure TfrmMainDml.actEditMyDictExecute(Sender: TObject);
var
S, fn: string;
begin
//20200406: 弃用
fn := 'MyDict.txt';
if Application.MessageBox(PChar(Format(srEzdmlConfirmEditTextFmt, [fn])),
PChar(Application.Title), MB_OKCANCEL or MB_ICONINFORMATION) <> idOk then
Exit;
S := GetFolderPathOfAppExe;
S := FolderAddFileName(S, fn);
if not FileExists(S) then
with TFileStream.Create(S, fmCreate) do
Free;
CtOpenDoc(PChar(S)); { *Converted from ShellExecute* }
if Application.MessageBox(PChar(Format(srEzdmlConfirmEditedTextFmt, [fn])),
PChar(Application.Title), MB_OKCANCEL or MB_ICONINFORMATION) <> idOk then
Exit;
GetAutoNameCapitalizer.ReloadDictFile;
end;
procedure TfrmMainDml.actEditSettingFileExecute(Sender: TObject);
var
S, fn: string;
//ws:WideString;
begin
//20200406: 弃用
//s:=trim('附件123');
//t:=IntToStr(Length(s)); {本身就是UTF8编码}
//t:=t+' '+Utf8ToAnsi(s)+':' +IntToStr(length(Utf8ToAnsi(s)));
//t:=t+' '+UTF8Decode(s)+':' +IntToStr(Length(UTF8Decode(s)));
fn := 'INI';
if Application.MessageBox(PChar(Format(srEzdmlConfirmEditTextFmt, [fn])),
PChar(Application.Title), MB_OKCANCEL or MB_ICONINFORMATION) <> idOk then
Exit;
S := GetConfFileOfApp;
if not FileExists(S) then
with TFileStream.Create(S, fmCreate) do
Free;
{$ifdef WINDOWS}
CtOpenDoc(PChar(S));
if Application.MessageBox(PChar(Format(srEzdmlConfirmEditedTextFmt, [fn])),
PChar(Application.Title), MB_OKCANCEL or MB_ICONINFORMATION) <> idOk then
Exit;
{$else}
RenameFile(S, S + '.txt');
try
CtOpenDoc(PChar(S + '.txt'));
if Application.MessageBox(PChar(Format(srEzdmlConfirmEditedTextFmt, [fn])),
PChar(Application.Title), MB_OKCANCEL or MB_ICONINFORMATION) <> idOk then
Exit;
finally
RenameFile(S + '.txt', S);
end;
{$endif}
LoadIni;
end;
procedure TfrmMainDml.actExecScriptExecute(Sender: TObject);
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actBatchOps.Execute;
end;
procedure TfrmMainDml.actExitExecute(Sender: TObject);
begin
Close;
end;
procedure TfrmMainDml.actExitWithoutSaveExecute(Sender: TObject);
begin
FCtDataModelList.Clear;
FFrameCtTableDef.Init(nil, True);
Close;
end;
procedure TfrmMainDml.actExportModelExecute(Sender: TObject);
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actExportXls.Execute;
end;
procedure TfrmMainDml.actEzdmlHomePageExecute(Sender: TObject);
var
S, V: string;
begin
EzdmlMenuActExecuteEvt('Help_EzdmlHome');
if not LangIsChinese then
S := 'http://www.ezdml.com/index.html'
else
S := 'http://www.ezdml.com/index_cn.html';
//S := 'http://blog.csdn.net/huzgd/';
V := Format(srEzdmlConfirmOpenUrlFmt, [S]);
if Application.MessageBox(PChar(V),
PChar(Application.Title), MB_OKCANCEL or MB_ICONWARNING) <> idOk then
Exit;
CtOpenDoc(PChar(S)); { *Converted from ShellExecute* }
end;
procedure TfrmMainDml.actFindObjectsExecute(Sender: TObject);
begin
// if not Assigned(Proc_ShowCtDmlSearch) then
// Exit;
//Proc_ShowCtDmlSearch(FGlobeDataModelList, nil);
if not FFrameCtTableDef.PanelDMLGraph.Visible then
Exit;
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actFindObject.Execute;
end;
procedure TfrmMainDml.actGenerateCodeExecute(Sender: TObject);
var
tbs: TCtMetaTableList;
begin
{$ifndef EZDML_LITE}
EzdmlMenuActExecuteEvt('Model_GenerateCode');
CheckCanEditMeta;
if not Assigned(frmCtGenCode) then
frmCtGenCode := TfrmCtGenCode.Create(Self);
frmCtGenCode.CtDataModelList := FCtDataModelList;
tbs := nil;
try
if FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameDMLGraph.GetSelectedTable <> nil then
begin
tbs := TCtMetaTableList.Create;
tbs.AutoFree := False;
FFrameCtTableDef.FFrameDMLGraph.CountSelectedTables(tbs);
if tbs.Count > 0 then
frmCtGenCode.MetaObjList := tbs;
end;
if frmCtGenCode.ShowModal = mrOk then
begin
end;
frmCtGenCode.CtDataModelList := FCtDataModelList;
finally
if tbs <> nil then
tbs.Free;
end;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actGenerateDatabaseExecute(Sender: TObject);
var
tbs: TCtMetaTableList;
begin
EzdmlMenuActExecuteEvt('Model_GenerateDatabase');
CheckCanEditMeta;
if not Assigned(frmCtGenSQL) then
frmCtGenSQL := TfrmCtGenSQL.Create(Self);
tbs := nil;
try
frmCtGenSQL.CtDataModelList := FCtDataModelList;
if FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameDMLGraph.GetSelectedTable <> nil then
begin
tbs := TCtMetaTableList.Create;
tbs.AutoFree := False;
FFrameCtTableDef.FFrameDMLGraph.CountSelectedTables(tbs);
if tbs.Count > 0 then
frmCtGenSQL.MetaObjList := tbs;
end;
frmCtGenSQL.SetWorkMode(0);
if frmCtGenSQL.ShowModal = mrOk then
begin
end;
frmCtGenSQL.CtDataModelList := FCtDataModelList;
finally
if tbs <> nil then
tbs.Free;
end;
end;
procedure TfrmMainDml.actGoTbFilterExecute(Sender: TObject);
begin
try
FFrameCtTableDef.FFrameCtTableList.edtTbFilter.SetFocus;
except
end;
end;
procedure TfrmMainDml.actImportDatabaseExecute(Sender: TObject);
var
C: integer;
begin
EzdmlMenuActExecuteEvt('Model_ImportDatabase');
CheckCanEditMeta;
if not Assigned(FfrmMetaImport) then
FfrmMetaImport := TfrmImportCtTable.Create(Self);
TfrmImportCtTable(FfrmMetaImport).FCtMetaObjList :=
Self.FCtDataModelList.CurDataModel.Tables;
TfrmImportCtTable(FfrmMetaImport).SetWorkMode(0);
C := Self.FCtDataModelList.CurDataModel.Tables.Count;
if FfrmMetaImport.ShowModal = mrOk then
begin
//FFrameCtTableDef.Init(FCtDataModelList, False);
FFrameCtTableDef.FFrameCtTableList.RefreshTheTree;
FFrameCtTableDef.RefreshProp;
if C = 0 then
if Self.FCtDataModelList.CurDataModel.Tables.Count > 2 then
begin
FFrameCtTableDef.FFrameDMLGraph.RearrangeAll;
end;
end;
end;
procedure TfrmMainDml.actModelOptionsExecute(Sender: TObject);
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actColorStyles.Execute;
end;
procedure TfrmMainDml.actNewFileExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('File_New');
if (GetKeyState(VK_SHIFT) and $80) <> 0 then
begin
WindowFuncs.CtOpenDoc(Application.ExeName);
Exit;
end;
CheckCanEditMeta;
FCtDataModelList.Pack;
if FCtDataModelList.Count = 0 then
Exit;
if Application.MessageBox(PChar(srEzdmlConfirmNewFile),
PChar(srEzdmlNew), MB_OKCANCEL or MB_ICONWARNING) <> idOk then
Exit;
if FCurFileName <> '' then
begin
if not FSaveTempFileOnExit then
begin
PromptSaveFile;
end
else
SaveDMLFastTmpFile(True);
end;
try
FCtDataModelList.Clear;
FFrameCtTableDef.Init(FCtDataModelList, True);
if FCtDataModelList.CurDataModel = nil then
Exit;
FFrameCtTableDef.Init(FCtDataModelList, False);
SetStatusBarMsg('');
FCurFileName := '';
FAutoSaveCounter := 0;
FCurDmlFileName := '';
TryLockFile('');
CheckCaption;
SaveIni;
finally
FCurFileName := '';
FAutoSaveCounter := 0;
FCurDmlFileName := '';
end;
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'FILE_NEW', '', Self, nil);
end;
end;
procedure TfrmMainDml.actNewModelExecute(Sender: TObject);
begin
FFrameCtTableDef.FFrameCtTableList.actNewModel.Execute;
end;
procedure TfrmMainDml.actNewTableExecute(Sender: TObject);
begin
//FFrameCtTableDef.FFrameCtTableList.actNewTable.Execute;
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actNewObj.Execute;
end;
procedure TfrmMainDml.actOpenFileExecute(Sender: TObject);
var
bDb: Boolean;
begin
EzdmlMenuActExecuteEvt('File_Open');
if (GetKeyState(VK_MENU) and $80) <> 0 then
begin
actOpenUrl.Execute;
Exit;
end;
bDb := False;
{if FCurFileName <> '' then
if IsDbFile(FCurFileName) then
bDb := True; }
if (GetKeyState(VK_SHIFT) and $80) <> 0 then
bDb := not bDb;
if bDb then
actLoadFromDb.Execute
else if OpenDialog1.Execute then
begin
PromptOpenFile(OpenDialog1.FileName);
end;
end;
procedure TfrmMainDml.actOpenLastFile1Execute(Sender: TObject);
var
fn: string;
begin
if FRecentFiles.Count > 0 then
try
fn := FRecentFiles[0];
if FCurFileName = fn then
begin
case Application.MessageBox(PChar(ExtractFileName(fn) + ' ' +
srEzdmlConfirmReOpenFile),
PChar(srEzdmlOpenFile), MB_YESNOCANCEL or MB_ICONWARNING) of
idYes:
begin
FCtDataModelList.Clear;
FFrameCtTableDef.Init(FCtDataModelList, True);
end
else
Exit;
end;
PromptOpenFile(fn, True);
end
else
PromptOpenFile(fn);
except
Application.HandleException(Self);
end;
end;
procedure TfrmMainDml.actRefreshExecute(Sender: TObject);
begin
FFrameCtTableDef.FFrameCtTableList.actRefresh.Execute;
end;
procedure TfrmMainDml.actSaveToDbExecute(Sender: TObject);
var
fs: TCtObjMemJsonSerialer;
po: Integer;
fn, S: string;
begin
EzdmlMenuActExecuteEvt('File_SaveToDb');
CheckCanEditMeta;
try
if not FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameCtTableList.TreeViewCttbs.CanFocus then
FFrameCtTableDef.FFrameCtTableList.TreeViewCttbs.SetFocus;
except
end;
FCtDataModelList.Pack;
frmEzdmlDbFile.Caption := actSaveToDb.Caption;
frmEzdmlDbFile.IsSaveMode := True;
S := Format(srNewDiagramNameFmt, [1]);
FCtDataModelList.Pack;
if FCtDataModelList.Count > 0 then
S := FCtDataModelList.Items[0].Name;
S := S+'_'+FormatDateTime('yyyymmdd_hhnn', Now);
if FCurFileName <> '' then
begin
S := FCurFileName;
if Self.IsTmpFile(S) then
begin
po := Pos('://', S);
if po>0 then
S:= Copy(S, po+3, Length(S));
S := ExtractFileName(S);
end;
end;
fn := S;
if fn <> '' then
begin
fn := ExtractFileName(fn);
fn := ChangeFileExt(fn,'');
end;
frmEzdmlDbFile.edtFileName.Text := fn;
if frmEzdmlDbFile.ShowModal <> mrOk then
begin
if FCurFileName <> '' then
begin
FAutoSaveCounter := 0;
if not FCtDataModelList.IsHuge then
SaveDmlToTmpFile;
end;
Exit;
end;
FProgressAll := 0;
FProgressCur := 0;
FWaitWnd := TfrmWaitWnd.Create(Self);
fs := TCtObjMemJsonSerialer.Create(False);
try
fs.RootName := 'DataModels';
FWaitWnd.Init(srEzdmlSaveFile + ' ' + frmEzdmlDbFile.ResultFileName, srEzdmlSaving,
srEzdmlAbortSaving);
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'DB_FILE_SAVE', frmEzdmlDbFile.ResultFileName, Self, nil);
end;
FCtDataModelList.SaveToSerialer(fs);
fs.EndJsonWrite;
fs.Stream.Seek(0, soFromBeginning);
if not frmEzdmlDbFile.SaveDataToDbFile(fs.Stream, frmEzdmlDbFile.ResultFileName, True) then
Exit;
frmEzdmlDbFile.ListViewFiles.Items.Clear;
FCurFileName := 'db://'+GetLastCtDbIdentStr+'/'+frmEzdmlDbFile.ResultFileName;
RememberFileDateSize;
finally
fs.Free;
FWaitWnd.Release;
FWaitWnd := nil;
end;
if not FCtDataModelList.IsHuge then
SaveDmlToTmpFile;
FAutoSaveCounter := 0;
CheckCaption;
S := Format(srEzdmlDbFileSavedFmt, [frmEzdmlDbFile.ResultFileName]);
SetStatusBarMsg(Format(srEzdmlDbFileSavedFmt, [frmEzdmlDbFile.ResultFileName]));
SetRecentFile(FCurFileName);
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'DB_FILE_SAVED', frmEzdmlDbFile.ResultFileName, Self, nil);
end;
if Application.MessageBox(PChar(S),
PChar(Application.Title),
MB_YESNOCANCEL or MB_ICONINFORMATION) <> idYes then
Exit;
actGenerateDatabaseExecute(nil);
end;
procedure TfrmMainDml.actSettingsExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('Tools_Settings');
if (GetKeyState(VK_SHIFT) and $80) <> 0 then
begin
actEditSettingFile.Execute;
Exit;
end;
if ShowEzdmlSettings(0) then
begin
LoadIni;
end;
end;
procedure TfrmMainDml.actShareFileExecute(Sender: TObject);
begin
FCtDataModelList.Pack;
with TfrmShareFile.Create(Application) do
try
InitDml(FCtDataModelList);
ShowModal;
finally
Free;
end;
end;
procedure TfrmMainDml.actShowDescTextExecute(Sender: TObject);
begin
if FFrameCtTableDef.FFrameCtTableProp.Showing then
FFrameCtTableDef.FFrameCtTableProp.actShowDescText.Execute;
end;
procedure TfrmMainDml.actShowHideListExecute(Sender: TObject);
begin
FFrameCtTableDef.ShowLeftTree := not FFrameCtTableDef.ShowLeftTree;
end;
procedure TfrmMainDml.actToggleTableViewExecute(Sender: TObject);
var
cto: TCtMetaObject;
begin
cto := FFrameCtTableDef.GetCurObject;
if cto is TCtDataModelGraph then
cto := nil;
if cto = nil then
if FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameCtTableList.CtTableList <> nil then
if FFrameCtTableDef.FFrameCtTableList.CtTableList.Count > 0 then
cto := FFrameCtTableDef.FFrameCtTableList.CtTableList.Items[0];
if cto = nil then
Exit;
FFrameCtTableDef.ShouldFocusUITick := GetTickCount64;
if FFrameCtTableDef.PanelDMLGraph.Visible then
begin
if cto is TCtMetaField then
cto := TCtMetaField(cto).OwnerTable;
FFrameCtTableDef.FFrameCtTableList.FocusToTable(cto.Name);
end
else
begin
FFrameCtTableDef.FFrameCtTableList.actFindInGraph.Execute;
FFrameCtTableDef.TryFocusGraph;
end;
end;
procedure TfrmMainDml.actEditGlobalScriptExecute(Sender: TObject);
var
S, fn: string;
begin
{$ifndef EZDML_LITE}
fn := DEF_GSCRIPT_FN;
S := GetFolderPathOfAppExe;
S := FolderAddFileName(S, fn);
if not FileExists(S) then
begin
if Application.MessageBox(PChar(Format(srEzdmlCreateGScriptTipFmt, [S])),
PChar(Application.Title),
MB_YESNOCANCEL or MB_ICONINFORMATION) <> idYes then
Exit;
with TStringList.Create do
try
Text := DEF_GSCRIPT_PS;
SaveToFile(S);
finally
Free;
end;
end;
fn := S;
if not Assigned(scriptIdeEditor) then
Application.CreateForm(TfrmScriptIDE, scriptIdeEditor);
with scriptIdeEditor do
begin
DmlScInit(fn, nil, nil, nil);
ed.ClearAll;
DmlScLoadFromFile(fn);
ed.Modified := False;
FileModified := False;
ActiveFile := fn;
ShowModal;
end;
CheckReloadGlobalScript;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actFullScreenExecute(Sender: TObject);
var
mon: TMonitor;
begin
EzdmlMenuActExecuteEvt('Model_FullScreen');
actFullScreen.Checked := not actFullScreen.Checked;
Self.FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actFullScreen.Checked :=
actFullScreen.Checked;
{$IFDEF UNIX}
if actFullScreen.Checked then
begin
ShowWindow(Handle, SW_SHOWFULLSCREEN);
Self.FFrameCtTableDef.SplitterCttbTree.Visible := False;
Self.FFrameCtTableDef.PanelCttbTree.Visible := False;
{$IFNDEF DARWIN}
Self.Menu := nil;
{$ENDIF}
Self.StatusBar1.Visible := False;
Refresh;
try
Self.FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.DMLGraph.SetFocus;
except
end;
end
else
begin
ShowWindow(Handle, SW_SHOWNORMAL);
;
{$IFNDEF DARWIN}
Self.Menu := Self.MainMenu1;
{$ENDIF}
Self.FFrameCtTableDef.PanelCttbTree.Visible := True;
Self.FFrameCtTableDef.SplitterCttbTree.Visible := True;
Self.StatusBar1.Visible := True;
end;
{$ELSE}
if actFullScreen.Checked then
begin
if WindowState <> wsNormal then
WindowState := wsNormal;
FFullScrnSaveBound := Self.BoundsRect;
mon := Screen.MonitorFromRect(FFullScrnSaveBound);
Self.FFrameCtTableDef.SplitterCttbTree.Visible := False;
Self.FFrameCtTableDef.PanelCttbTree.Visible := False;
Self.Menu := nil;
//Self.FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.ToolBar1.Visible := False;
Self.StatusBar1.Visible := False;
Self.BorderStyle := bsNone;
//Self.FormStyle := fsStayOnTop;
Self.Left := mon.Left;
Self.Top := mon.Top;
Self.ClientWidth := mon.Width;
Self.ClientHeight := mon.Height;
Refresh;
try
Self.FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.DMLGraph.SetFocus;
except
end;
end
else
begin
//Self.FormStyle := fsNormal;
Self.BorderStyle := bsSizeable;
Self.Menu := Self.MainMenu1;
Self.FFrameCtTableDef.PanelCttbTree.Visible := True;
Self.FFrameCtTableDef.SplitterCttbTree.Visible := True;
//Self.FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.ToolBar1.Visible := True;
Self.StatusBar1.Visible := True;
Self.BoundsRect := FFullScrnSaveBound;
end;
{$ENDIF}
if FFrameCtTableDef.PanelCttbTree.Visible then
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actShowHideList.ImageIndex := FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actShowHideList.Tag;
end
else
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actShowHideList.ImageIndex := FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actShowHideList.Tag + 1;
end;
end;
procedure TfrmMainDml.actGenerateLastCodeExecute(Sender: TObject);
begin
{$ifndef EZDML_LITE}
EzdmlMenuActExecuteEvt('Model_GenerateLastCode');
CheckCanEditMeta;
if not Assigned(frmCtGenCode) then
frmCtGenCode := TfrmCtGenCode.Create(Self);
frmCtGenCode.CtDataModelList := FCtDataModelList;
frmCtGenCode.TimerAutoGen.Tag := 1;
if frmCtGenCode.ShowModal = mrOk then
begin
end;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actGenerateTestDataExecute(Sender: TObject);
var
tbs: TCtMetaTableList;
begin
{$ifndef EZDML_LITE}
EzdmlMenuActExecuteEvt('Model_GenerateTestData');
CheckCanEditMeta;
if not Assigned(frmCtGenData) then
frmCtGenData := TfrmCtGenData.Create(Self);
frmCtGenData.CtDataModelList := FCtDataModelList;
tbs := nil;
try
if FFrameCtTableDef.PanelDMLGraph.Visible then
if FFrameCtTableDef.FFrameDMLGraph.GetSelectedTable <> nil then
begin
tbs := TCtMetaTableList.Create;
tbs.AutoFree := False;
FFrameCtTableDef.FFrameDMLGraph.CountSelectedTables(tbs);
if tbs.Count > 0 then
frmCtGenData.MetaObjList := tbs;
end;
if frmCtGenData.ShowModal = mrOk then
begin
end;
frmCtGenData.CtDataModelList := FCtDataModelList;
finally
if tbs <> nil then
tbs.Free;
end;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actHttpServerExecute(Sender: TObject);
begin
{$ifndef EZDML_LITE}
EzdmlMenuActExecuteEvt('Tools_HttpServer');
if not Assigned(FfrmHttpServer) then
FfrmHttpServer := TfrmHttpSvr.Create(Self);
FfrmHttpServer.ShowModal;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actImportDDLSqlExecute(Sender: TObject);
var
oc: Integer;
begin
{$ifndef EZDML_LITE}
with TfrmDDLSqlImp.Create(Self) do
try
FCtTbList := Self.FCtDataModelList.CurDataModel.Tables;
FCtTbList.Pack;
oc := FCtTbList.Count;
FCtDataModelList.ModelFileConfig.LastModel := '';
if ShowModal = mrOk then
begin
FFrameCtTableDef.FFrameCtTableList.RefreshTheTree;
FFrameCtTableDef.RefreshProp;
if oc=0 then
if FCtTbList.Count > 2 then
begin
FCtDataModelList.ModelFileConfig.LastModel := Self.FCtDataModelList.CurDataModel.Name;
TimerDelayCmd.Tag := 11;
TimerDelayCmd.Enabled := True;
end;
end;
finally
Release;
end;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actImportExcelExecute(Sender: TObject);
begin
{$ifndef EZDML_LITE}
with TfrmExcelImp.Create(Self) do
try
FCtTbList := Self.FCtDataModelList.CurDataModel.Tables;
if ShowModal = mrOk then
begin
FFrameCtTableDef.FFrameCtTableList.RefreshTheTree;
FFrameCtTableDef.RefreshProp;
end;
finally
Release;
end;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actCharCodeToolExecute(Sender: TObject);
begin
{$ifndef EZDML_LITE}
if FFindHexDlg = nil then
FFindHexDlg := TfrmFindHex.Create(Self);
FFindHexDlg.ShowModal;
{$else}
raise Exception.Create(srEzdmlLiteNotSupportFun);
{$endif}
end;
procedure TfrmMainDml.actChatGPTExecute(Sender: TObject);
begin
CallAI(1);
end;
procedure TfrmMainDml.actCheckUpdatesExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('Help_CheckUpdates');
CheckForUpdates(True);
CheckShowNewVersionInfo(True);
end;
procedure TfrmMainDml.actImportFileExecute(Sender: TObject);
begin
if OpenDialogImp.Execute then
begin
PromptOpenFile(OpenDialogImp.FileName);
end;
end;
procedure TfrmMainDml.actLoadFromDbExecute(Sender: TObject);
var
fn: String;
begin
CheckCanEditMeta;
if FFileWorking then
Exit;
if Assigned(FWaitWnd) then
raise Exception.Create('wait wnd busy');
FFileWorking := True;
try
frmEzdmlDbFile.Caption := actLoadFromDb.Caption;
frmEzdmlDbFile.IsSaveMode:=False;
if frmEzdmlDbFile.ShowModal <> mrOk then
Exit;
fn := 'db://'+GetLastCtDbIdentStr+'/'+frmEzdmlDbFile.ResultFileName;
finally
FFileWorking := False;
end;
PromptOpenFile(fn);
if FCurFileName = fn then
frmEzdmlDbFile.CheckLockAfterOpen;
end;
procedure TfrmMainDml.actNewAppWinExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('File_NewEzdmlWin');
WindowFuncs.CtOpenDoc(Application.ExeName);
end;
procedure TfrmMainDml.actOpenUrlExecute(Sender: TObject);
procedure SaveToTmpFile(const url, sid, data: string);
var
sfn, fn, dir: string;
ts: TStringList;
begin
sfn := GetUrlParamVal(Url,'cap');
if sfn='' then
sfn := sid;
sfn := 'web://'+sfn;
fn := GetNewTmpFileName(sfn);
fn := ChangeFileExt(fn,'.~dmj');
dir := ExtractFilePath(fn);
if not DirectoryExists(dir) then
ForceDirectories(dir);
ts:= TStringList.Create;
try
ts.Text := data;
ts.SaveToFile(fn);
finally
ts.Free;
end;
Self.LoadFromFile(fn);
AddOnlineHistoryFile(sid,url,'',Length(data));
//FCurFileName := sfn;
//FCurDmlFileName := sfn;
//CheckCaption;
end;
var
sid, Url, rurl, S, sfn: String;
po: Integer;
begin
EzdmlMenuActExecuteEvt('File_OpenURL');
if frmOnlineFile=nil then
frmOnlineFile := TfrmOnlineFile.Create(Application);
if frmOnlineFile.ShowModal <> mrOk then
Exit;
Url := frmOnlineFile.edtFileURL.Text;
sid := GetUrlParamVal(Url,'sid');
if sid='' then
raise Exception.Create('SID not found - '+URL);
sfn := GetUrlParamVal(Url,'cap');
if sfn='' then
sfn := sid;
CheckCanEditMeta;
FCtDataModelList.Pack;
if FCtDataModelList.TableCount > 0 then
case Application.MessageBox(PChar(ExtractFileName(sfn) + ' ' +
srEzdmlConfirmClearOnOpen),
PChar(srEzdmlOpenFile), MB_OKCANCEL or MB_ICONWARNING) of
idOk:
if FCurFileName <> '' then
begin
if not FSaveTempFileOnExit then
begin
PromptSaveFile;
end
else
SaveDMLFastTmpFile(True);
end;
idNo:
begin
Exit;
//vOldMds := TCtDataModelGraphList.Create;
//vOldMds.AssignFrom(FCtDataModelList);
end
else
Exit;
end;
rurl := 'http://ezdml.com/ez/mdown/?sid='+sid+'&uid='+GetMyComputerId;
S := GetUrlData_Net(rurl, '', '[SHOW_PROGRESS]');
if FFileWorking then
Exit;
if Pos('[Error]', Trim(S))=1 then
begin
S:=Trim(S);
S:=Copy(S,8,Length(S));
raise Exception.Create(S);
end;
SaveToTmpFile(url, sid, S);
end;
procedure TfrmMainDml.actQuickStartExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('Help_QuickStart');
if frmHelpAbout = nil then
begin
frmHelpAbout := TfrmHelpAbout.Create(Self);
frmHelpAbout.LoadFile('');
end;
frmHelpAbout.ShowModal;
end;
procedure TfrmMainDml.actRestoreDatabaseExecute(Sender: TObject);
begin
if not Assigned(frmCtGenSQL) then
frmCtGenSQL := TfrmCtGenSQL.Create(Self);
frmCtGenSQL.MetaObjList := nil;
frmCtGenSQL.SetWorkMode(1);
if frmCtGenSQL.LoadDbBackFile then
if frmCtGenSQL.ShowModal = mrOk then
begin
end;
end;
procedure TfrmMainDml.actSaveFileAsExecute(Sender: TObject);
var
bSaveUCodeJson: boolean;
po: Integer;
S: String;
begin
EzdmlMenuActExecuteEvt('File_SaveAs');
CheckCanEditMeta;
S := Format(srNewDiagramNameFmt, [1]);
FCtDataModelList.Pack;
if FCtDataModelList.Count > 0 then
S := FCtDataModelList.Items[0].Name;
S := S+'_'+FormatDateTime('yyyymmdd_hhnn', Now);
if FCurFileName <> '' then
begin
S := FCurFileName;
if Self.IsTmpFile(S) then
begin
po := Pos('://', S);
if po>0 then
S:= Copy(S, po+3, Length(S));
S := ExtractFileName(S);
end;
end;
SaveDialog1.FileName := S;
if SaveDialog1.Execute then
begin
bSaveUCodeJson := stringsAsUtf8Encode;
if (GetKeyState(VK_CONTROL) and $80) <> 0 then
if LowerCase(ExtractFileExt(SaveDialog1.FileName)) = '.dmj' then
case Application.MessageBox(PChar(srEzdmlDmjUnicodePropmt),
PChar(Application.Title),
MB_YESNOCANCEL or MB_ICONQUESTION) of
idYes:
stringsAsUtf8Encode := False;
idNo:
stringsAsUtf8Encode := True;
else
Exit;
end;
try
TryLockFile(SaveDialog1.FileName);
SaveToFile(SaveDialog1.FileName);
TryLockFile(FCurFileName);
finally
stringsAsUtf8Encode := bSaveUCodeJson;
end;
FAutoSaveCounter := 0;
SaveDmlToTmpFile;
SetStatusBarMsg(srEzdmlSaved + GetStatusPanelFileName(FCurFileName) + ' ' + TimeToStr(Now));
FCurDmlFileName := FCurFileName;
SetRecentFile(FCurFileName);
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'FILE_SAVE', FCurDmlFileName, Self, nil);
end;
end;
end;
procedure TfrmMainDml.actSaveFileExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('File_Save');
CheckCanEditMeta;
if (FCurFileName <> '') and not IsTmpFile(FCurFileName) and not IsDbFile(FCurFileName) then
begin
if (GetKeyState(VK_SHIFT) and $80) <> 0 then
begin
actSaveToDb.Execute;
Exit;
end;
SaveToFile(FCurFileName);
FAutoSaveCounter := 0;
if not FCtDataModelList.IsHuge then
SaveDmlToTmpFile;
SetStatusBarMsg(srEzdmlSaved + GetStatusPanelFileName(FCurFileName) + ' ' + TimeToStr(Now));
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'FILE_SAVE', FCurDmlFileName, Self, nil);
end;
end
else if Pos('db://', FCurFileName) = 1 then
begin
if (GetKeyState(VK_SHIFT) and $80) <> 0 then
begin
actSaveFileAs.Execute;
Exit;
end;
actSaveToDb.Execute;
end
else
begin
if (GetKeyState(VK_SHIFT) and $80) <> 0 then
begin
actSaveToDb.Execute;
Exit;
end;
actSaveFileAs.Execute;
end;
end;
procedure TfrmMainDml.actShowFileInExplorerExecute(Sender: TObject);
var
fn: string;
begin
fn := FCurFileName;
if (fn = '') or not FileExists(fn) then
fn := Application.ExeName;
CtBrowseFile(fn);
end;
procedure TfrmMainDml.actAboutEzdmlExecute(Sender: TObject);
begin
EzdmlMenuActExecuteEvt('Help_About');
with TfrmAbout.Create(Self) do
try
ShowModal;
finally
Free;
end;
end;
procedure TfrmMainDml.actBackupDatabaseExecute(Sender: TObject);
begin
if not Assigned(FfrmMetaImport) then
FfrmMetaImport := TfrmImportCtTable.Create(Self);
TfrmImportCtTable(FfrmMetaImport).FCtMetaObjList := nil;
TfrmImportCtTable(FfrmMetaImport).SetWorkMode(1);
if FfrmMetaImport.ShowModal = mrOk then
begin
//FFrameCtTableDef.Init(FCtDataModelList, False);
FFrameCtTableDef.FFrameCtTableList.RefreshTheTree;
end;
end;
procedure TfrmMainDml.actBrowseCustomToolsExecute(Sender: TObject);
var
dir: string;
begin
dir := GetFolderPathOfAppExe('CustomTools');
if not DirectoryExists(dir) then
dir := GetFolderPathOfAppExe('');
CtOpenDir(dir);
end;
procedure TfrmMainDml.actBrowseScriptsExecute(Sender: TObject);
var
dir: string;
begin
dir := GetFolderPathOfAppExe('Templates');
if not DirectoryExists(dir) then
dir := GetFolderPathOfAppExe('');
CtOpenDir(dir);
end;
procedure TfrmMainDml.actShowTmprFileExecute(Sender: TObject);
var
dir, fn: string;
begin
fn := FCurFileName;
if fn = '' then
if FCtDataModelList.TableCount > 0 then
fn := GetConfFileOfApp('.dmh');
dir := GetTmpDirForFile(fn);
if not DirectoryExists(dir) then
begin
dir := GetAppDefTempPath;
end;
if not DirectoryExists(dir) then
ForceDirectories(dir);
CtOpenDir(dir);
end;
procedure TfrmMainDml.actSqlToolExecute(Sender: TObject);
begin
ShowSqlEditor;
end;
procedure TfrmMainDml.actTogglePhyViewExecute(Sender: TObject);
begin
FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.actShowPhyView.Execute;
end;
procedure TfrmMainDml.CheckCaption;
begin
if FCurFileName = '' then
begin
Caption := FOrginalCaption;
Application.Title := srEzdmlAppTitle;
end
else
begin
Caption := FOrginalCaption + ' - ' + FCurFileName;
if IsDbFile(FCurFileName) and not FFileDbConnectOk then
Caption := Caption +' '+ srEzdmlDbOfflineTip;
Application.Title := srEzdmlAppTitle + ' - ' + ExtractFileName(FCurFileName);
end;
end;
function TfrmMainDml.CheckCurFileDateSizeChanged: boolean;
var
sz, st: integer;
vFileDate: TDateTime;
begin
Result := False;
if FCurFileName = '' then
Exit;
if IsTmpFile(FCurFileName) then
Exit;
if (FCurFileSize = 0) and (FCurFileDate = 0) then
Exit;
if IsDbFile(FCurFileName) then
begin
sz := 0;
vfileDate := Now;
st := frmEzdmlDbFile.CheckDbFileState(FCurFileName, sz, vfileDate, False);
if st <= 2 then
begin
if FFileDbConnectOk then
begin
FFileDbConnectOk := False;
CheckCaption;
if ShowMessageOnTop(srEzdmlPromptDbFileDisconnected, Application.Title)=IDOK then
begin
if ExecCtDbLogon >= 0 then
Result := CheckCurFileDateSizeChanged;
end;
end;
Exit;
end;
if not FFileDbConnectOk then
begin
FFileDbConnectOk := True;
CheckCaption;
end
else
FFileDbConnectOk := True;
if sz <> Self.FCurFileSize then
begin
Result := True;
Exit;
end;
if Abs(vFileDate - FCurFileDate) > 2 / 24 / 60 / 60 then
begin
Result := True;
Exit;
end;
Exit;
end;
if not FileExists(FCurFileName) then
Exit;
sz := GetDocFileSize(FCurFileName);
if sz <> Self.FCurFileSize then
begin
Result := True;
Exit;
end;
if GetDmlFileDate(FCurFileName, vFileDate) then
begin
if Abs(vFileDate - FCurFileDate) > 2 / 24 / 60 / 60 then
begin
Result := True;
Exit;
end;
end;
end;
procedure TfrmMainDml.PromptOpenFile(fn: string; bDisableTmpFiles: boolean);
function HasDbTmpFile: Boolean;
var
S: String;
begin
Result := False;
S := GetFastTmpFileName(fn);
if S = '' then
Exit;
if not FileExists(S) then
Exit;
Result := True;
end;
procedure LoadDbTmpFile;
var
S: String;
begin
S := GetFastTmpFileName(fn);
if S = '' then
raise Exception.Create(Format(srEzdmlFileNotFoundFmt, [fn]));
if not FileExists(S) then
raise Exception.Create(Format(srEzdmlFileNotFoundFmt, [S]));
LoadFromFile(S);
end;
procedure LoadFromDFile(dfn: string);
begin
if IsDbFile(dfn) then
begin
LoadFromDbFile(dfn);
end
else
LoadFromFile(dfn);
end;
var
vOldMds: TCtDataModelGraphList;
I: integer;
dbTmp: Boolean;
begin
if FFileWorking then
Exit;
CheckCanEditMeta;
FCtDataModelList.Pack;
if FCtDataModelList.TableCount > 0 then
case Application.MessageBox(PChar(ExtractFileName(fn) + ' ' +
srEzdmlConfirmClearOnOpen),
PChar(srEzdmlOpenFile), MB_OKCANCEL or MB_ICONWARNING) of
idOk:
if FCurFileName <> '' then
begin
if not FSaveTempFileOnExit then
begin
PromptSaveFile;
end
else
SaveDMLFastTmpFile(True);
end;
idNo:
begin
Exit;
//vOldMds := TCtDataModelGraphList.Create;
//vOldMds.AssignFrom(FCtDataModelList);
end
else
Exit;
end;
dbTmp := False;
if IsDbFile(fn) then
begin
I := CheckDbFileState(fn, True);
if I < 2 then
begin
if not IsTmpFile(fn) and not bDisableTmpFiles then
begin
if HasDbTmpFile then
begin
if Application.MessageBox(PChar(fn + ' ' +
srEzdmlConfirmOpenDbTmpFile),
PChar(srEzdmlOpenFile), MB_OKCANCEL or MB_ICONWARNING) <> IDOK then
Abort;
dbTmp := True;
end
else
Abort;
end
else
Abort;
end;
if I = 2 then
begin
RemoveRecentFile(fn);
raise Exception.Create(Format(srEzdmlFileNotFoundFmt, [fn]));
end;
end
else if not FileExists(fn) then
begin
RemoveRecentFile(fn);
raise Exception.Create(Format(srEzdmlFileNotFoundFmt, [fn]));
end;
TryLockFile(fn);
vOldMds := nil;
try
if not IsDbFile(fn) then
if LowerCase(ExtractFileExt(fn)) = '.pdm' then
begin
ImportFromFile(fn);
Exit;
end;
if not IsTmpFile(fn) and not bDisableTmpFiles then
begin
if dbTmp then
LoadDbTmpFile
else if TryLoadFromTmpFile(fn) then
begin
end
else
LoadFromDFile(fn);
end
else
LoadFromDFile(fn);
if not dbTmp then
FCurDmlFileName := fn;
if Assigned(vOldMds) then
begin
for I := 0 to vOldMds.Count - 1 do
FCtDataModelList.NewModelItem.AssignFrom(vOldMds[I]);
FFrameCtTableDef.Init(FCtDataModelList, False);
end;
if not dbTmp then
SetRecentFile(fn);
finally
if Assigned(vOldMds) then
vOldMds.Free;
end;
if Assigned(GProc_OnEzdmlCmdEvent) then
begin
GProc_OnEzdmlCmdEvent('MAINFORM', 'FILE_OPEN', FCurDmlFileName, Self, nil);
end;
end;
procedure TfrmMainDml.PromptSaveFile;
begin
case Application.MessageBox(PChar(srEzdmlPromptSaveFile), PChar(Application.Title),
MB_YESNOCANCEL or MB_ICONWARNING) of
idYes: ;
idNo:
Exit;
else
Abort;
end;
if (IsTmpFile(FCurFileName) or (FCurFileName = '')) then
begin
actSaveFileAs.Execute;
if (IsTmpFile(FCurFileName) or (FCurFileName = '')) then
begin
Abort;
end;
end
else
SaveToFile(FCurFileName);
end;
procedure TfrmMainDml.ReCreateCustomToolsMenu;
function GetCustomToolsDir: string;
begin
Result := GetFolderPathOfAppExe('CustomTools');
end;
function GetCustomToolsFiles: string;
var
Sr: TSearchRec;
AFolderName: string;
I: Integer;
begin
Result := '';
{$ifdef EZDML_LITE}
for I := 0 to High(CtPsLiteRegs) do
if CtPsLiteRegs[I].Cat='Tool' then
Result := Result + CtPsLiteRegs[I].Name + #13#10;
{$endif}
AFolderName := GetCustomToolsDir;
if not DirectoryExists(AFolderName) then
Exit;
if FindFirst(FolderAddFileName(AFolderName, '*.*'),
SysUtils.faAnyFile,
//SysUtils.faAnyFile + SysUtils.faHidden + SysUtils.faSysFile + SysUtils.faDirectory + SysUtils.faArchive,
Sr) = 0 then
try
repeat
if (Sr.Name = '.') or (Sr.Name = '..') then
Continue;
if (Sr.Attr and SysUtils.faDirectory) <> 0 then
Continue
else
Result := Result + SR.Name + #13#10;
until FindNext(Sr) <> 0;
finally
FindClose(Sr);
end;
Result := Trim(Result);
end;
var
mn: TMenuItem;
I: integer;
fn: string;
begin
if FReservedToolsMenuCount = 0 then
FReservedToolsMenuCount := MnTools1.Count;
FCustomTools.Text := GetCustomToolsFiles;
for I := MnTools1.Count - 1 downto FReservedToolsMenuCount do
MnTools1.Items[I].Free;
for I := 0 to FCustomTools.Count - 1 do
begin
fn := FCustomTools[I];
mn := TMenuItem.Create(Self);
mn.Caption := ChangeFileExt(fn, '');
mn.Hint := fn;
mn.Tag := I;
mn.OnClick := _OnCustomToolsClick;
if I < 8 then
begin
mn.ShortCut := TextToShortCut('Alt+' + IntToStr(I + 2));
end;
MnTools1.Add(mn);
end;
end;
procedure TfrmMainDml.RecreateRecentMn;
procedure RecreateRecentMnEx(Pmns: TMenuItem);
var
mn: TMenuItem;
I: integer;
fn: string;
begin
Pmns.Clear;
for I := 0 to FRecentFiles.Count - 1 do
begin
fn := FRecentFiles[I];
mn := TMenuItem.Create(Self);
if IsDbFile(fn) then
mn.Caption := '@'+ExtractDmlFileName(fn)
else
mn.Caption := ExtractFileName(fn);
mn.Hint := fn;
mn.Tag := I;
mn.OnClick := _OnRecentFileClick;
Pmns.Add(mn);
end;
end;
begin
RecreateRecentMnEx(MN_Recentfiles);
RecreateRecentMnEx(FFrameCtTableDef.FFrameDMLGraph.FFrameCtDML.MNOF_Recentfiles);
end;
procedure TfrmMainDml.RememberFileDateSize;
var
vFileSize: Integer;
vFileDate: TDateTime;
begin
if FCurFileName = '' then
Exit;
if IsTmpFile(FCurFileName) then
Exit;
FCurFileSize := 0;
FCurFileDate := Now;
if not GetDmlFileDateAndSize(FCurFileName, vFileSize, vFileDate) then
Exit;
FCurFileSize := vfileSize;
FCurFileDate := vfileDate;
end;
procedure TfrmMainDml.RemoveRecentFile(fn: string);
var
I, idx: integer;
ini: TIniFile;
S: string;
begin
if fn = '' then
Exit;
S := LowerCase(fn);
idx := -1;
for I := 0 to FRecentFiles.Count - 1 do
if LowerCase(FRecentFiles[I]) = S then
begin
idx := I;
Break;
end;
if idx<0 then
Exit;
FRecentFiles.Delete(idx);
ini := TIniFile.Create(GetConfFileOfApp);
try
ini.EraseSection('RecentFiles');
for I := 0 to FRecentFiles.Count - 1 do
ini.WriteString('RecentFiles', IntToStr(I + 1), FRecentFiles[I]);
ini.WriteString('RecentFiles', 'CurFileName', FCurFileName);
finally
ini.Free;
end;
RecreateRecentMn;
end;
initialization
G_CreateSeqForOracle := False;
G_GenSqlSketchMode := False;
G_BackupBeforeAlterColumn := False;
G_BigIntForIntKeys := False;
G_QuotReservedNames := False;
G_QuotAllNames := False;
G_LogicNamesForTableData := False;
G_AddLimitToCursorSQL := True;
G_MaxRowCountForTableData := 25;
G_CtSqlMaxFetchCount := 1000;
G_WriteConstraintToDescribeStr := True;
G_FieldGridShowLines := True;
G_AddColCommentToCreateTbSql := True;
G_CreateForeignkeys := True;
G_CreateIndexForForeignkey := False;
G_HiveVersion := 2;
G_MysqlVersion := 5;
G_AutoCommit := True;
G_RetainAfterCommit := False;
G_ShowJdbcConsole := False;
G_EnableCustomPropUI := False;
G_EnableAdvTbProp := False;
G_EnableTbPropGenerate := True;
G_EnableTbPropRelations := True;
G_EnableTbPropData := False;
G_EnableTbPropUIDesign := False;
G_TableDialogViewModeByDefault := False;
G_CheckForUpdates := True;
Proc_CheckStringMaxLen := CheckStringMaxLen;
Proc_CheckCustDataTypeReplaces := CheckCustDataTypeReplaces;
Proc_OnExecAppCmd := EzdmlExecAppCmd;
Proc_CreateCtObjSerialer := EzdmlCreateCtObjSerialer;
Proc_GetSelectedCtMetaObj := ezdml_GetSelectedCtMetaObj;
Proc_JsonPropProc := Ezdml_JsonPropProc;
Proc_CtObjToJsonStr := CtObjToJsonStr; //added by huz 20210214
Proc_ReadCtObjFromJsonStr := ReadCtObjFromJsonStr;
{$ifndef EZDML_LITE}
Proc_GenDemoData := CtGenTestData;
Proc_GetTableDemoDataJson := CtGenTableDemoDataJson;
{$endif}
Proc_CheckDecDmlData := Ezdml_CheckDecDmlData;
InitCtChnNames;
finalization
if Assigned(G_Reserved_Keywords) then
FreeAndNil(G_Reserved_Keywords);
end.
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。