Files
HeidiSQL/source/updatecheck.pas

213 lines
6.6 KiB
ObjectPascal

unit updatecheck;
interface
uses
Windows, Messages, SysUtils, Classes, Forms,
Dialogs, StdCtrls, ExtActns, IniFiles, Controls, Graphics, Registry;
type
TfrmUpdateCheck = class(TForm)
btnCancel: TButton;
groupBuild: TGroupBox;
btnBuild: TButton;
groupRelease: TGroupBox;
btnRelease: TButton;
lblStatus: TLabel;
memoRelease: TMemo;
memoBuild: TMemo;
procedure FormCreate(Sender: TObject);
procedure btnBuildClick(Sender: TObject);
procedure btnReleaseClick(Sender: TObject);
procedure FormShow(Sender: TObject);
private
{ Private declarations }
CheckfileDownload : TDownLoadURL;
ReleaseURL, BuildURL : String;
procedure Status(txt: String);
procedure ReadCheckFile;
public
{ Public declarations }
AutoClose: Boolean; // Automatically close dialog after detecting no available downloads
CurrentRevision: Integer;
end;
implementation
uses helpers, main;
{$R *.dfm}
{$I const.inc}
{**
Set defaults
}
procedure TfrmUpdateCheck.FormCreate(Sender: TObject);
begin
// Should be false by default. Callers can set this to True after Create()
AutoClose := False;
end;
{**
Update status text
}
procedure TfrmUpdateCheck.Status(txt: String);
begin
lblStatus.Caption := txt;
Repaint;
end;
{**
Download check file
}
procedure TfrmUpdateCheck.FormShow(Sender: TObject);
var
TempPath: array[0..MAX_PATH] of Char;
reg : TRegistry;
begin
Status('Initiating ... ');
Caption := 'Check for '+APPNAME+' updates ...';
CurrentRevision := StrToIntDef(AppRevision, 0);
// Init GUI controls
btnRelease.Enabled := False;
btnBuild.Enabled := False;
memoRelease.Clear;
memoBuild.Clear;
// Detect temp directory
GetTempPath(MAX_PATH, @TempPath);
// Prepare download
CheckfileDownload := TDownLoadURL.Create(Self);
CheckfileDownload.URL := APPDOMAIN + 'updatecheck.php';
CheckfileDownload.Filename := StrPas(TempPath) + APPNAME + '_updatecheck.ini';
// Download the check file
Screen.Cursor := crHourglass;
try
Status('Downloading check file ...');
CheckfileDownload.ExecuteTarget(nil);
Status('Reading check file ...');
ReadCheckFile;
if (not groupRelease.Enabled) and (not groupBuild.Enabled) then begin
// Developer versions probably have "unknown" (0) as revision,
// which makes it impossible to compare the revisions.
if CurrentRevision = 0 then
Status('Error: Cannot determine current revision. Using a developer version?')
else
Status('Your '+APPNAME+' is up-to-date (no update available).');
end else
Status('Updates available.');
// Remember when we did the updatecheck to enable the automatic interval
reg := TRegistry.Create;
reg.OpenKey(REGPATH, true);
reg.WriteString(REGNAME_LAST_UPDATECHECK, DateTimeToStr(Now));
reg.CloseKey;
FreeAndNil(reg);
except
// Do not popup errors, just display them in the status label
On E:Exception do
Status(E.Message);
end;
if FileExists(CheckfileDownload.Filename) then
DeleteFile(CheckfileDownload.Filename);
FreeAndNil(CheckfileDownload);
Screen.Cursor := crDefault;
// For automatic updatechecks this dialog should close if no updates are available.
// Using PostMessage, as Self.Close or ModalResult := mrCancel does not work
// as expected in FormShow
if AutoClose and (not groupRelease.Enabled) and (not groupBuild.Enabled) then
PostMessage(Self.Handle, WM_CLOSE, 0, 0);
end;
{**
Parse check file for updated version + release
}
procedure TfrmUpdateCheck.ReadCheckFile;
var
Ini : TIniFile;
ReleaseVersion : String;
ReleaseRevision, BuildRevision : Integer;
Note : String;
Compiled : TDateTime;
const
INISECT_RELEASE = 'Release';
INISECT_BUILD = 'Build';
begin
// Read [Release] section of check file
Ini := TIniFile.Create(CheckfileDownload.Filename);
if Ini.SectionExists(INISECT_RELEASE) then begin
ReleaseVersion := Ini.ReadString(INISECT_RELEASE, 'Version', 'unknown');
ReleaseRevision := Ini.ReadInteger(INISECT_RELEASE, 'Revision', 0);
ReleaseURL := Ini.ReadString(INISECT_RELEASE, 'URL', '');
memoRelease.Lines.Add( 'Version ' + ReleaseVersion + ' (yours: '+AppVersion+')' );
memoRelease.Lines.Add( 'Released: ' + Ini.ReadString(INISECT_RELEASE, 'Date', '') );
Note := Ini.ReadString(INISECT_RELEASE, 'Note', '');
if Note <> '' then
memoRelease.Lines.Add( 'Note: ' + Note );
btnRelease.Caption := 'Download version ' + ReleaseVersion;
// Enable the download button if the current version is outdated
groupRelease.Enabled := (CurrentRevision > 0) and (ReleaseRevision > CurrentRevision);
btnRelease.Enabled := groupRelease.Enabled;
memoRelease.Enabled := groupRelease.Enabled;
if not memoRelease.Enabled then
memoRelease.Font.Color := cl3DDkShadow
else
memoRelease.Font.Color := clWindowText;
end;
// Read [Build] section of check file
if Ini.SectionExists(INISECT_BUILD) then begin
BuildRevision := Ini.ReadInteger(INISECT_BUILD, 'Revision', 0);
BuildURL := Ini.ReadString(INISECT_BUILD, 'URL', '');
memoBuild.Lines.Add( 'Revision ' + IntToStr(BuildRevision) + ' (yours: '+AppRevision+')' );
FileAge(ParamStr(0), Compiled);
memoBuild.Lines.Add( 'Compiled: ' + Ini.ReadString(INISECT_BUILD, 'Date', '') + ' (yours: '+DateToStr(Compiled)+')' );
Note := Ini.ReadString(INISECT_BUILD, 'Note', '');
if Note <> '' then
memoBuild.Lines.Add( 'Note: ' + Note );
btnBuild.Caption := 'Download build ' + IntToStr(BuildRevision);
// A new release should have priority over a new nightly build.
// So the user should not be able to download a newer build here
// before having installed the new release.
groupBuild.Enabled := (CurrentRevision > 0) and (BuildRevision > CurrentRevision) and (not btnRelease.Enabled);
btnBuild.Enabled := groupBuild.Enabled;
memoBuild.Enabled := groupBuild.Enabled;
if not memoBuild.Enabled then
memoBuild.Font.Color := cl3DDkShadow
else
memoBuild.Font.Color := clWindowText;
end;
end;
{**
Download release installer via web browser
}
procedure TfrmUpdateCheck.btnReleaseClick(Sender: TObject);
begin
ShellExec(ReleaseURL);
end;
{**
Download latest build via web browser
TODO: internally replace heidisql.exe:
1. create a batch file which replaces heidisql.exe
2. quit Heidi
3. start the batch
4. start Heidi
}
procedure TfrmUpdateCheck.btnBuildClick(Sender: TObject);
begin
ShellExec(BuildURL);
end;
end.