OSDN Git Service

[YCへマージ]
[winbottle/winbottle.git] / bottleclient / MainForm.pas
index 33a22a6..e4ace7d 100755 (executable)
@@ -174,8 +174,17 @@ type
     mnUndo: TMenuItem;
     mnRedo: TMenuItem;
     N9: TMenuItem;
-    ScriptBackUp: TTimer;
     mnPresetReplaceRoot: TMenuItem;
+    ScriptBackup: TTimer;
+    OpenDialog: TOpenDialog;
+    mnFileOpen: TMenuItem;
+    actFileOpen: TAction;
+    actFileSaveAs: TAction;
+    mnSaveAs: TMenuItem;
+    SaveDialog: TSaveDialog;
+    N11: TMenuItem;
+    mnSave: TMenuItem;
+    actFileSave: TAction;
     procedure actConfirmExecute(Sender: TObject);
     procedure FormCreate(Sender: TObject);
     procedure FormDestroy(Sender: TObject);
@@ -247,6 +256,7 @@ type
     procedure actDownloadLogExecute(Sender: TObject);
     procedure cbxTargetGhostChange(Sender: TObject);
     procedure actFMOExplorerExecute(Sender: TObject);
+    // \8eÀ\8dÛ\82Ì\8f\88\97\9d\82ÍLogInsertCue\82Ö\88Ú\93®
     procedure actInsertCueExecute(Sender: TObject);
     procedure FormResize(Sender: TObject);
     procedure actCopyExecute(Sender: TObject);
@@ -270,17 +280,17 @@ type
     procedure actUndoExecute(Sender: TObject);
     procedure actRedoExecute(Sender: TObject);
 //    procedure IdSLPP20ConnectFailed(Sender: TObject);
-    procedure ScriptBackUpTimer(Sender: TObject);
-    //DispatchBottle()\82Ö\83o\83C\83p\83X
-    procedure BottleCnv(Mid, Channel, Ghost, Script: String;
-      Vote, Agree: integer; LogTime: TDateTime);
+    procedure ScriptBackupTimer(Sender: TObject);
+    procedure actFileOpenExecute(Sender: TObject);
+    procedure actFileSaveAsExecute(Sender: TObject);
+    procedure actFileSaveExecute(Sender: TObject);
   private
     FSleeping: boolean;  // \94z\91\97\83X\83\8a\81[\83v\92\86\82©\82Ç\82¤\82©
     FStatusText: String;
 //    FConnecting: boolean;
 //    FAdded: boolean;
     FBooted: boolean; //\8f\89\89ñ\8bN\93®\92Ê\90M\97p
-    FEndSession: Boolean; // Windows\8fI\97¹\82ð\8c\9f\92m\82µ\82Ätrue\82É\82È\82é
+//    FEndSession: Boolean; // Windows\8fI\97¹\82ð\8c\9f\92m\82µ\82Ätrue\82É\82È\82é
     FOriginalCaption: String;
 //    FAutoAddAfterGetChannel: boolean; //\83`\83\83\83\93\83l\83\8b\8eæ\93¾\8cã\82É\83_\83C\83A\83\8d\83O\82È\82µ\82É
                                       //\83`\83\83\83\93\83l\83\8b\82É\8eQ\89Á\82·\82é\82©\82Ç\82¤\82©
@@ -290,11 +300,16 @@ type
 //    FNowChannel: String; //\8c»\8dÝ\91I\91ð\82³\82ê\82Ä\82¢\82é\83`\83\83\83\93\83l\83\8b
 //    JoinChannelsBackup: TStringList; //\88ê\8e\9e\8eg\97p
     //
-    FScriptModified: boolean; // \83X\83N\83\8a\83v\83g\82ª\95Ï\8dX\82³\82ê\82Ä\82¢\82é\82©\82Ç\82¤\82©\81B
-                              // \83\8d\81[\83J\83\8b\8am\94F\8b­\90§\97p\83t\83\89\83O\81BTRichEdit.Modified\82Í
-                              //\95Ê\82Ì\97p\93r\82Å\8eg\82Á\82Ä\82¢\82é\82Ì\82Å\8eg\82¦\82È\82¢
+//    FScriptModified: boolean; // \83X\83N\83\8a\83v\83g\82ª\95Ï\8dX\82³\82ê\82Ä\82¢\82é\82©\82Ç\82¤\82©\81B
+//                              // \83\8d\81[\83J\83\8b\8am\94F\8b­\90§\97p\83t\83\89\83O\81BTRichEdit.Modified\82Í
+//                              //\95Ê\82Ì\97p\93r\82Å\8eg\82Á\82Ä\82¢\82é\82Ì\82Å\8eg\82¦\82È\82¢
+    //
+    FScriptBackupModified: boolean; // \83X\83N\83\8a\83v\83g\83o\83b\83N\83A\83b\83v\83t\83\89\83O
+    FScriptBackupFile: string;      // \8c»\8dÝ\82Ì\83o\83b\83N\83A\83b\83v\83t\83@\83C\83\8b\96¼
+    FScriptBackupDir: string;       // \83o\83b\83N\83A\83b\83v\90æ\82Ì\83t\83H\83\8b\83_\96¼
     //
-    FScriptBackModified: boolean; //\83X\83N\83\8a\83v\83g\8e©\93®\95Û\91\83t\83\89\83O
+    FFileModified: boolean;       //\83t\83@\83C\83\8b\93à\97e\82ª\95Ï\8dX\82³\82ê\82Ä\82¢\82é\82©\81B
+    FFileName: string;            //\90V\8bK\82©\8aù\91\83t\83@\83C\83\8b\82Ì\94»\92f\81A\82Ü\82½\82Í\8aù\91\83t\83@\83C\83\8b\82Ì\8fê\8f\8a
     //
 //    FDragTabIndex: integer; //\83^\83u\83h\83\89\83b\83O\83h\83\8d\83b\83v\8aÖ\98A
 //    FDragTabDest: integer;  //\83h\83\8d\83b\83v\82·\82é\88Ê\92u(\82·\82®\89E\82É\82­\82é\83^\83u\82Ì\83C\83\93\83f\83b\83N\83X)
@@ -366,14 +381,26 @@ type
     procedure ClearEditor;
     procedure CopyFromLogToEditor(Log: TLogItem);
     //
-    procedure AppendTextLog(const FileName, Line: String);
+    procedure AppendTextLog(const FileName, Line: String; FAppend: boolean);
     procedure AppendXMLLog(const FileName: String; Args: THeadValue);
     //\8e©\93®\95Û\91\90Ý\92è
-    procedure SetScriptAutoBackUp;
+    procedure SetScriptAutoBackup;
+    //\92P\91Ì\83t\83@\83C\83\8b\93ü\8fo\97Í\8aÖ\8cW
+    procedure PerformFileOpen(const AFileName: string);
+    procedure PerformFileSave(const AFileName: string);
+    procedure SetFileName(const FileName: string; Reset, SetNewFile: boolean);
+    function CheckFileModified(Sender: TObject): integer;
+    function FileSave(Sender: TObject): integer;
+    function FileSaveAs(Sender: TObject): integer;
+    procedure EditerStatusChange;
+    procedure SetFileModified(Value: boolean);
+    function CheckFileExt(const FileName: string): boolean;
+    procedure DeleteBackupFile(const FileName: string);
+
   protected
     procedure WndProc(var Message: TMessage); override;
-    procedure WMQueryEndSession(var msg: TWMQueryEndSession);
-      message WM_QUERYENDSESSION;
+//    procedure WMQueryEndSession(var msg: TWMQueryEndSession);
+//      message WM_QUERYENDSESSION;
   public
     function DoTrans(var Script: String;
       Options: TScriptTransOptions): String; overload;
@@ -392,6 +419,12 @@ type
 //    procedure PostCommand(Command: TStrings); overload;
 //    procedure PostSetChannel(Channels: TStrings);
     procedure SaveChainRuleList;
+    //DispatchBottle()\82Ö\83o\83C\83p\83X
+    procedure BottleCnv(Log: TLogItem);
+    //\83u\83\89\83E\83U\94»\92f\82ÆURL\88ø\93n\82µ
+    procedure OpenBrowser(const Url: string);
+    //Log\83E\83B\83\93\83h\83E\82©\82ç\8fð\8c\8f\95t\82Å\8cÄ\82Î\82ê\82é
+    procedure LogInsertCue(TestAction, SelectAll: boolean);
   end;
 
 
@@ -406,13 +439,16 @@ const
   PanelStatus     = 4;  //SSTP Bottle\83T\81[\83o\82É\93o\98^\82³\82ê\82Ä\82¢\82Ü\82·\81c\82È\82Ç
 
   IconConnected    = 17;
-  IconDisconnected = 47;  //\8b\8c18  \8e¸\94s\8fó\91Ô\82ð\8fí\82É\8eg\97p
+  IconDisconnected = 18;
   IconSleep        = 19;
-  IconSleepDisconnected = 47; //\8b\8c20
+  IconSleepDisconnected = 20;
+  IconYC                = 47;
 
   WarningColor = clRed;
 
   SendButtonLongHint = 'Bottle\82Ì\91\97\90M';
+  NewFileTitle       = '\96¼\8fÌ\96¢\92è';     // \90V\8bK\83^\83C\83g\83\8b
+  StdBackupFile      = 'Script.bak';   // FileCheck\8ew\92è\82Ì\8ag\92£\8eq\82ð\8eg\82¤
 
 function Token(const Str: String; const Delimiter: char;
   const Index: integer): String;
@@ -423,7 +459,7 @@ implementation
 
 uses SettingForm, LogForm,
   MessageBox, FMOExplorer, EditorTalkShow;
-//SendConfirm, ChannelListForm, 
+//SendConfirm, ChannelListForm,
 
 {$R *.DFM}
 
@@ -514,7 +550,7 @@ begin
 
   if cbxTargetGhost.ItemIndex > 0 then
     AGhost := cbxTargetGhost.Text
-{
+{ \95K\82¸\8ew\92è\82³\82ê\82é\82Í\82¸
   else if FNowChannel <> '' then
     AGhost := ChannelList.Channel[FNowChannel].Ghost
 }
@@ -538,7 +574,7 @@ begin
     Item.Free;
   end;
 
-  FScriptModified := false;
+//  FScriptModified := false;
 end;
 
 procedure TfrmSender.FormCreate(Sender: TObject);
@@ -553,6 +589,7 @@ begin
   Spps.LoadFromDirectory(FSppDir);
   ConstructMenu(false);
   BuildReplaceMenu(mnPresetReplaceRoot);
+  FScriptBackupDir := ExtractFileDir(Application.ExeName) + '\temp\';
 
   Str := TStringList.Create;
   try
@@ -689,10 +726,12 @@ end;
 
 // \83\81\83b\83Z\81[\83W\91\97\90M
 procedure TfrmSender.actSendExecute(Sender: TObject);
-var Talk, Ghost: String;
-//    Command: TStringList;
-    Err: String;
-//    F: TextFile;
+var
+  Talk, AGhost: String;
+//  Command: TStringList;
+  Err: String;
+//  F: TextFile;
+  Item: TLogItem;
 begin
   //\83t\83\89\83O
   if not Pref.SendAction then
@@ -700,7 +739,8 @@ begin
     ShowHintLabel('\93®\8dì\82ª\96¢\90Ý\92è\82Å\82·');
     Exit;
   end;
-  if Length(GetScriptText) = 0 then begin
+  if Length(GetScriptText) = 0 then
+  begin
     ShowMessage('\83X\83N\83\8a\83v\83g\82ª\8bó\82Å\82·\81B');
     Exit;
   end;
@@ -725,7 +765,8 @@ begin
   YenETrans;
   Talk := StringReplace(GetScriptText, #13#10, '', [rfReplaceAll]);
   Err := DoTrans(Talk, [toWarnMessySurface, toWarnCheck]);
-  if Err <> '' then begin
+  if Err <> '' then
+  begin
     MessageDlg(Err, mtWarning, [mbOk], 0);
     Exit;
   end;
@@ -742,10 +783,25 @@ begin
   end;
 }
 
-  Ghost := '';
-  if cbxTargetGhost.ItemIndex > 0 then Ghost := cbxTargetGhost.Text;
+  AGhost := '';
+  if cbxTargetGhost.ItemIndex > 0 then AGhost := cbxTargetGhost.Text;
   //\92P\91Ì\83A\83N\83V\83\87\83\93
-  BottleCnv('', 'local', Ghost, Talk, 0, 0, now());
+  Item := TLogItem.Create;
+  try
+    with Item do
+    begin
+      MID := '';
+      Channel := 'local';
+      LogTime := now();
+      Script := Talk;
+      Ghost := AGhost;
+      Votes := 0;
+      Agrees := 0;
+    end;
+    BottleCnv(Item);
+  except
+    Item.Free;
+  end;
   ShowHintLabel('[local]\82Ö\91\97\90M\82µ\82Ü\82µ\82½');
 
 {
@@ -838,7 +894,8 @@ begin
     try
       HeadValue := THeadValue.Create(Str);
     except
-      ShowMessage('SSTP Bottle\83T\81[\83o\82ª\89ð\90Í\82Å\82«\82È\82¢\83G\83\89\81[\82ð\95Ô\82µ\82Ü\82µ\82½\81B');
+      Beep;
+      frmMessageBox.ShowMessage('SSTP Bottle\83T\81[\83o\82ª\89ð\90Í\82Å\82«\82È\82¢\83G\83\89\81[\82ð\95Ô\82µ\82Ü\82µ\82½\81B');
       Exit;
     end;
     Command := HeadValue['Command'];
@@ -846,11 +903,11 @@ begin
     if ResStr = 'Err' then begin
       if HeadValue['ExtraMessage'] <> '' then begin
         Beep;
-        ShowMessage('SSTP Bottle\83T\81[\83o\82ª\8e\9f\82Ì\83G\83\89\81[\82ð\95Ô\82µ\82Ü\82µ\82½:'#13#10 +
+        frmMessageBox.ShowMessage('SSTP Bottle\83T\81[\83o\82ª\8e\9f\82Ì\83G\83\89\81[\82ð\95Ô\82µ\82Ü\82µ\82½:'#13#10 +
                      HeadValue['ExtraMessage']);
       end else begin
         Beep;
-        ShowMessage('SSTP Bottle\83T\81[\83o\82ª\89½\82ç\82©\82Ì\83G\83\89\81[\82ð\95Ô\82µ\82Ü\82µ\82½\81B');
+        frmMessageBox.ShowMessage('SSTP Bottle\83T\81[\83o\82ª\89½\82ç\82©\82Ì\83G\83\89\81[\82ð\95Ô\82µ\82Ü\82µ\82½\81B');
       end;
     end;
     if (Command = 'sendBroadcast') and (ResStr = 'OK') then begin
@@ -923,7 +980,7 @@ begin
     if (Command = 'setChannels') then begin
       if ResStr <> 'OK' then begin
         Beep;
-        ShowMessage('\83`\83\83\83\93\83l\83\8b\90Ý\92è\82É\8e¸\94s\82µ\82Ü\82µ\82½\81B\82à\82¤\88ê\93x\93o\98^\82µ\82È\82¨\82µ\82Ä\82­\82¾\82³\82¢');
+        frmMessageBox.ShowMessage('\83`\83\83\83\93\83l\83\8b\90Ý\92è\82É\8e¸\94s\82µ\82Ü\82µ\82½\81B\82à\82¤\88ê\93x\93o\98^\82µ\82È\82¨\82µ\82Ä\82­\82¾\82³\82¢');
         ShowHintLabel('\83`\83\83\83\93\83l\83\8b\90Ý\92è\82É\8e¸\94s\82µ\82Ü\82µ\82½', WarningColor);
       end;
     end;
@@ -958,11 +1015,8 @@ begin
   ShowHintLabel('SSTP Bottle\83T\81[\83o\82É\82Í\90Ú\91±\82Å\82«\82Ü\82¹\82ñ', WarningColor);
 {
   // \8b­\90§\8dÄ\90Ú\91±\82ð\8ds\82¤
-  IdSlpp20.OnDisconnect := nil;
-  if IdSlpp20.Connected then IdSlpp20.Disconnect;
-  FAutoAddAfterGetChannel := true;
-  BeginConnect;
-  IdSlpp20.OnDisconnect := Slpp20Disconnect;
+  FBeginConnectFailCount := 0; // \8e©\93®\8dÄ\90Ú\91±\83J\83E\83\93\83^\83\8a\83Z\83b\83g
+  RetryBeginConnect;
 }
 end;
 
@@ -976,15 +1030,14 @@ begin
   FAutoAddAfterGetChannel := Pref.AutoStart;
 }
   FBooted := true;
-  frmLog.Show;
+  if NOT Pref.NotShowLog then frmLog.Show;  // \8bN\93®\8e\9e\82É\83\8d\83O\83E\83B\83\93\83h\83E\82ð\95\\8e¦\82·\82é\82©
   frmSurfacePreview.Show;
   Self.Show;
   SakuraSeeker.BeginDetect;
   SakuraSeekerDetectResultChanged(self);
-  //\8e©\93®\95Û\91\90Ý\92è\82Ì\83\8d\81[\83h
-  SetScriptAutoBackUp;
+  SetScriptAutoBackup;                      // \8e©\93®\95Û\91\90Ý\92è\82Ì\83\8d\81[\83h
 {
-  if SakuraSeeker.Count = 0 then
+  if (SakuraSeeker.Count = 0) and (Pref.NotWarnGhostIsntExists = False) then
     frmMessageBox.ShowMessage('\83S\81[\83X\83g(SSTP\83T\81[\83o)\82ª1\82Â\82à\8bN\93®\82µ\82Ä\82¢\82Ü\82¹\82ñ\81B'#13#10 +
       'SSTP Bottle\82ð\97\98\97p\82·\82é\82½\82ß\82É\82Í\81A\83S\81[\83X\83g\82ð\93¯\8e\9e\82É\8bN\93®\82µ\82Ä\82­\82¾\82³\82¢\81B'#13#10 +
       '\8fÚ\8d×\82Í\83w\83\8b\83v\82ð\82²\97\97\89º\82³\82¢\81B');
@@ -1016,20 +1069,33 @@ begin
     actSendToLogWindow.Execute
   else
     ClearEditor;
-  FScriptBackModified := false; //\8e©\93®\95Û\91\88ê\8e\9e\92â\8e~
 end;
 
 procedure TfrmSender.memScriptChange(Sender: TObject);
+begin
+  //\83G\83f\83B\83^\93à\97e\82ª\95Ï\8dX\82³\82ê\82½\82Æ\82«
+  EditerStatusChange;             // \83X\83e\81[\83^\83X\8dX\90V\82Ö
+  FScriptBackupModified := true;  // \8e©\93®\95Û\91\8dÄ\8aJ
+  SetFileModified(true);          // \83t\83@\83C\83\8b\95Ï\8dX\83t\83\89\83O
+end;
+
+procedure TfrmSender.EditerStatusChange;
 var
   Script: String;
   Text: String;
 begin
+  // \83X\83e\81[\83^\83X\95Ï\8dX
+  // \93à\97e\82ª\83h\83\89\83b\83O\82³\82ê\82½\82Æ\82«\82à\8cÄ\82Ñ\8fo\82³\82ê\82é
   Script := StringReplace(GetScriptText, #13#10, '', [rfReplaceAll]);
-  Text := Format('%d\83o\83C\83g/%d\95b', [Length(Script), SsPlayTime.PlayTime(Script) div 1000]);
+  if Pref.NotScriptTime then
+    // \83o\83C\83g\90\94\82¾\82¯\95\\8e¦
+    Text := Format('%d\83o\83C\83g', [Length(Script)])
+  else
+    // \83o\83C\83g\90\94\81{\97\\91ª\8e\9e\8aÔ\95\\8e¦
+    Text := Format('%d\83o\83C\83g/%d\95b', [Length(Script), SsPlayTime.PlayTime(Script) div 1000]);
   StatusBar.Panels[PanelBytes].Text := Text;
-  FScriptModified := true;
+//  FScriptModified := true;
   EditorPreview;
-  FScriptBackModified := true; //\8e©\93®\95Û\91\8dÄ\8aJ
 end;
 
 procedure TfrmSender.mnStayOnTopClick(Sender: TObject);
@@ -1117,7 +1183,7 @@ end;
 
 procedure TfrmSender.ChangeTaskIcon;
 var Ico: TIcon;
-    IcoNum: integer;
+//    IcoNum: integer;
 begin
 {
   if Added then begin
@@ -1127,13 +1193,10 @@ begin
     else IcoNum := IconDisconnected;
   end;
 }
-  //\83^\83X\83N\83A\83C\83R\83\93\82Ì\8fó\91Ô\81i\8e¸\94s\8fó\91Ô\8eg\97p\81j
-  if Sleeping then IcoNum := IconSleepDisconnected
-    else IcoNum := IconDisconnected;
   try
     Ico := TIcon.Create;
     try
-      imgIcon.GetIcon(IcoNum, Ico);
+      imgIcon.GetIcon(IconYC, Ico);  // IconNum
       TaskTray.Icon := Ico;
       TaskTray.Registered := true;
     finally
@@ -1161,7 +1224,7 @@ begin
   begin
     StatusBar.Panels[PanelStatus].Text := GetLongHint(Application.Hint);
     Application.HintColor := clInfoBk;
-{
+{ \91\97\90M\83{\83^\83\93\8aÖ\98A
     if (Application.Hint = SendButtonLongHint)
     and (FNowChannel <> '') then
     begin
@@ -1190,6 +1253,9 @@ begin
   //ConstMenuBar.Menu := nil;
   ConstMenuBar.AutoSize := false;
   ConstMenuBar.Width := 1000;
+  // \83\81\83j\83\85\81[\82ð\88ê\93x\8fÁ\82·
+  // \82È\82º\82©2\89ñ\96Ú\88È\8d~\82Ì\95\\8e¦\82ª\82¨\82©\82µ\82­\82È\82é
+  ConstMenuBar.Menu := nil;
   ConstMenuBar.Menu := ConstBarMenu;
   ConstMenuBar.AutoSize := true;
 end;
@@ -1428,7 +1494,7 @@ end;
 
 procedure TfrmSender.mnGoToHPClick(Sender: TObject);
 begin
-  ShellExecute(Handle, 'open', PChar(Pref.HomePage), nil, nil, SW_SHOW);
+  OpenBrowser(Pref.HomePage);
 end;
 
 procedure TfrmSender.ShowHintLabel(const Mes: String; Col: TColor);
@@ -1477,15 +1543,24 @@ begin
         //\83\81\83b\83Z\81[\83W\8eó\90M
         DispatchBottle(EventType, HeadValue);
       end;
+
       etMemberCount: begin
+        //\91\8d\8eQ\89Á\8eÒ\90\94
         StatusBar.Panels[PanelMembers].Text := HeadValue['Num'] + '\90l'
       end;
+
       etChannelCount: begin
+        //\83`\83\83\83\93\83l\83\8b\95Ê\8eQ\89Á\8eÒ\90\94
         try
-          ChannelList.Channel[HeadValue['Channel']].Members := StrToInt(HeadValue['Num']);
+          if HeadValue['Channel'] <> '' then begin
+            if ChannelList.Channel[HeadValue['Channel']] <> nil then begin
+              ChannelList.Channel[HeadValue['Channel']].Members := StrToInt(HeadValue['Num']);
+            end;
+          end;
         except
         end;
       end;
+
       etConnectOk: begin
         ShowHintLabel('SSTP Bottle\83T\81[\83o\82Æ\92Ê\90M\8am\97§\81B');
         Added := true;
@@ -1493,34 +1568,50 @@ begin
         //\83`\83\83\83\93\83l\83\8b\8e©\93®\93o\98^
         if not Connecting then
           PostCommand(['Command: getChannels']);
-        SakuraSeeker.BeginDetect;
+        try
+          SakuraSeeker.BeginDetect;
+        except
+          on E: Exception do ShowHintLabel(E.Message,WarningColor);
+        end;
       end;
+
       etChannelList: begin
         UpdateJoinChannelList(HeadValue);
         // \8dÅ\8cã\82É\8eQ\89Á\82µ\82Ä\82¢\82½\83`\83\83\83\93\83l\83\8b\82ð\8bL\98^\82·\82é
         if JoinChannelsBackup = nil then JoinChannelsBackup := TStringList.Create;
         JoinChannelsBackup.Assign(JoinChannels);
       end;
+
       etCloseChannel: begin
-        with JoinChannels do
-          if IndexOf(HeadValue['Channel']) >= 0 then
-            Delete(IndexOf(HeadValue['Channel']));
-        with tabChannel do begin
-          if Tabs.IndexOf(HeadValue['Channel']) >= 0 then
-            Tabs.Delete(Tabs.IndexOf(HeadValue['Channel']));
-          if Tabs.Count > 0 then TabIndex := 0 else TabIndex := -1;
-          tabChannelChange(self);
+        //\83`\83\83\83\93\83l\83\8b\94p\8e~
+        if HeadValue['Channel'] <> '' then begin
+          with JoinChannels do
+            if IndexOf(HeadValue['Channel']) >= 0 then
+              Delete(IndexOf(HeadValue['Channel']));
+          with tabChannel do begin
+            if Tabs.IndexOf(HeadValue['Channel']) >= 0 then
+              Tabs.Delete(Tabs.IndexOf(HeadValue['Channel']));
+            if Tabs.Count > 0 then TabIndex := 0 else TabIndex := -1;
+            tabChannelChange(self);
+          end;
+          ShowHintLabel(HeadValue['Channel'] + '\83`\83\83\83\93\83l\83\8b\82Í\94p\8e~\82³\82ê\82Ü\82µ\82½',
+                        WarningColor);
+          frmLog.AddCurrentSystemLog('SYSTEM', HeadValue['Channel'] + '\83`\83\83\83\93\83l\83\8b\82Í\94p\8e~\82³\82ê\82Ü\82µ\82½');
+          frmMessageBox.ShowMessage(HeadValue['Channel'] + '\83`\83\83\83\93\83l\83\8b\82Í\94p\8e~\82³\82ê\82Ü\82µ\82½');
         end;
-        ShowHintLabel(HeadValue['Channel'] + '\83`\83\83\83\93\83l\83\8b\82Í\94p\8e~\82³\82ê\82Ü\82µ\82½',
-                      WarningColor);
-        frmLog.AddCurrentSystemLog('SYSTEM', HeadValue['Channel'] + '\83`\83\83\83\93\83l\83\8b\82Í\94p\8e~\82³\82ê\82Ü\82µ\82½');
-        frmMessageBox.ShowMessage(HeadValue['Channel'] + '\83`\83\83\83\93\83l\83\8b\82Í\94p\8e~\82³\82ê\82Ü\82µ\82½');
       end;
+
       etForceBroadcastInformation: begin
-        if HeadValue['Type'] = 'Vote' then begin
-          frmLog.VoteLog(HeadValue['MID'], StrToIntDef(HeadValue['Num'], 0));
-        end else if HeadValue['Type'] = 'Agree' then begin
-          frmLog.AgreeLog(HeadValue['MID'], StrToIntDef(HeadValue['Num'], 0));
+        //\93\8a\95[\81^\93¯\88Ó\81^\82»\82Ì\91¼\83u\83\8d\81[\83h\83L\83\83\83X\83g\8fî\95ñ
+        if HeadValue['MID'] <> '' then begin
+          try //Num\82ª\90\94\92l\82Å\82È\82©\82Á\82½\82Æ\82«\91Î\8dô
+            if HeadValue['Type'] = 'Vote' then begin
+              frmLog.VoteLog(HeadValue['MID'], StrToIntDef(HeadValue['Num'], 0));
+            end else if HeadValue['Type'] = 'Agree' then begin
+              frmLog.AgreeLog(HeadValue['MID'], StrToIntDef(HeadValue['Num'], 0));
+            end;
+          except
+          end;
         end;
       end;
     end;
@@ -1543,7 +1634,10 @@ begin
 end;
 
 procedure TfrmSender.actSettingExecute(Sender: TObject);
+var
+  FTBackupMode: boolean;
 begin
+  FTBackupMode := Pref.ScriptBackupMode;
   Application.CreateForm(TfrmSetting, frmSetting);
   try
     frmSetting.Execute;
@@ -1558,8 +1652,12 @@ begin
   UpdateLayout;
   tabChannel.Repaint;
   frmLog.UpdateWindow;
-  SetScriptAutoBackUp;  //\83X\83N\83\8a\83v\83g\8e©\93®\95Û\91\90Ý\92è\8dX\90V
+  SetScriptAutoBackup;  //\83X\83N\83\8a\83v\83g\8e©\93®\95Û\91\90Ý\92è\8dX\90V
   UpdateChannelList;    //\83`\83\83\83\93\83l\83\8b\83S\81[\83X\83g\8dÄ\83\8d\81[\83h
+
+  // \83o\83b\83N\83A\83b\83v\83\82\81[\83h\82ª\95Ï\8dX\82³\82ê\82½\82Æ\82«FScriptBackupFile\82à\95Ï\8dX\82·\82é
+  if FTBackupMode <> Pref.ScriptBackupMode then
+    SetFileName(FFileName, false, false);
 end;
 
 procedure TfrmSender.memScriptKeyPress(Sender: TObject; var Key: Char);
@@ -1678,9 +1776,8 @@ end;
 
 {
 procedure TfrmSender.UpdateJoinChannelList(Dat: THeadValue);
-var
-  i: integer;
-  nodat: boolean;
+var i: integer;
+    nodat: boolean;
 begin
   nodat := Dat = nil; //nil\82È\82ç\83`\83\83\83\93\83l\83\8b\91S\89ð\8f\9c
   if nodat then Dat := THeadValue.Create('');
@@ -1697,8 +1794,9 @@ begin
     Tabs.Clear;
     for i := 0 to JoinChannels.Count-1 do begin
       //\8eó\90M\90ê\97p\83`\83\83\83\93\83l\83\8b\82Í\95\\8e¦\82µ\82È\82¢
-      if not ChannelList.Channel[JoinChannels[i]].NoPost then
-        Tabs.Add(JoinChannels[i]);
+      if ChannelList.Channel[JoinChannels[i]] <> nil then
+        if not ChannelList.Channel[JoinChannels[i]].NoPost then
+          Tabs.Add(JoinChannels[i]);
     end;
     Tabs.EndUpdate;
     // \8c³\82©\82ç\83`\83\83\83\93\83l\83\8b\82É\8eQ\89Á\82µ\82Ä\82¢\82½\8fê\8d\87\82Í
@@ -1776,10 +1874,11 @@ var Opt: TSstpSendOptions;
     CueItem: TLogItem;
     ReplaceHash: THeadValue;
 begin
+  Channel := Dat['Channel'];
+
   Opt := [];
   if Pref.NoTranslate then Opt := Opt + [soNoTranslate];
   if Pref.NoDescript  then Opt := Opt + [soNoDescript];
-  Channel := Dat['Channel'];
 {
   case EventType of
     etScript: Sender := Channel;
@@ -1804,13 +1903,13 @@ begin
   ReplaceHash := THeadValue.Create;
   ReplaceHash['%channel%'] := SafeFileName(Dat['Channel']);
   ReplaceHash['%ghost%'] := SafeFileName(Dat['IfGhost']);
-  ReplaceHash['%date%'] := FormatDateTime('yy-mm-dd', Now());
-  ReplaceHash['%year%'] := FormatDateTime('yyyy', Now());
-  ReplaceHash['%yy%'] := FormatDateTime('yy', Now());
-  ReplaceHash['%month%'] := FormatDateTime('mm', Now());
-  ReplaceHash['%day%'] := FormatDateTime('dd', Now());
-  ReplaceHash['%hour%'] := FormatDateTime('hh', Now());
-  ReplaceHash['%minute%'] := FormatDateTime('nn', Now());
+  ReplaceHash['%date%'] := FormatDateTime('yy-mm-dd', LogTime);
+  ReplaceHash['%year%'] := FormatDateTime('yyyy', LogTime);
+  ReplaceHash['%yy%'] := FormatDateTime('yy', LogTime);
+  ReplaceHash['%month%'] := FormatDateTime('mm', LogTime);
+  ReplaceHash['%day%'] := FormatDateTime('dd', LogTime);
+  ReplaceHash['%hour%'] := FormatDateTime('hh', LogTime);
+  ReplaceHash['%minute%'] := FormatDateTime('nn', LogTime);
 
   Event := TBottleChainBottleEvent.Create;
   try
@@ -1875,7 +1974,7 @@ begin
           if Action is TBottleChainSaveTextLogAction then
             AppendTextLog(StringReplaceEx((Action as TBottleChainSaveTextLogAction).FileName, ReplaceHash),
               Format('%s,%s,%s,%s', [Dat['Channel'], Dat['IfGhost'],
-                FormatDateTime('yy/mm/dd hh:nn:ss', Now), Dat['Script']]));
+                FormatDateTime('yy/mm/dd hh:nn:ss', LogTime), Dat['Script']]), true);
           if Action is TBottleChainSaveXMLLogAction then
             AppendXMLLog(StringReplaceEx((Action as TBottleChainSaveXMLLogAction).FileName, ReplaceHash),
               Dat);
@@ -1894,7 +1993,7 @@ begin
         end else begin
           Ghost := Dat['TargetGhost']; // \83I\81[\83o\81[\83\89\83C\83h\82³\82ê\82Ä\82¢\82é\89Â\94\\90«\82ª\82 \82é
           CueItem := TLogItem.Create(ltBottle, Dat['MID'], Dat['Channel'],
-            Script, Ghost, Now());
+            Script, Ghost, LogTime);
           try
             FBottleSstp.Push(CueItem);
           except
@@ -1907,7 +2006,7 @@ begin
       if Dat['DialogMessage'] <> '' then begin
         Beep;
         frmMessageBox.ShowMessage(
-          DateTimeToStr(Now) + #13#10 +
+          DateTimeToStr(LogTime) + #13#10 +
           'SSTP Bottle\83T\81[\83o\82©\82ç\82¨\92m\82ç\82¹'#13#10+Dat['DialogMessage']);
         for i := 0 to LogNameList.Count-1 do
           frmLog.AddCurrentSystemLog(LogNameList[i], Dat['DialogMessage']);
@@ -1929,6 +2028,7 @@ procedure TfrmSender.YenETrans;
 var St, Le, i: integer;
     Orig, Text: String;
 begin
+  if Pref.NotYenETrans then exit; // \e\83`\83F\83b\83N\82µ\82È\82¢
   St := memScript.SelStart;
   Le := memScript.SelLength;
   Orig := GetScriptText;
@@ -2105,7 +2205,7 @@ end;
 
 procedure TfrmSender.mnGotoVoteClick(Sender: TObject);
 begin
-  ShellExecute(Handle, 'open', PChar(Pref.VotePage), nil, nil, SW_SHOW);
+  OpenBrowser(Pref.VotePage);
 end;
 
 procedure TfrmSender.tabChannelMouseMove(Sender: TObject;
@@ -2124,7 +2224,7 @@ end;
 
 procedure TfrmSender.mnGoToHelpClick(Sender: TObject);
 begin
-  ShellExecute(Handle, 'open', PChar(Pref.HelpPage), nil, nil, SW_SHOW);
+  OpenBrowser(Pref.HelpPage);
 end;
 
 procedure TfrmSender.tabChannelMouseDown(Sender: TObject;
@@ -2252,11 +2352,33 @@ end;
 procedure TfrmSender.FormCloseQuery(Sender: TObject;
   var CanClose: Boolean);
 begin
+{
   if (not Pref.ConfirmOnExit) or FEndSession then
+  begin
     Exit;
-  if MessageDlg('Yasagure Client\82ð\8fI\97¹\82µ\82Ü\82·', mtConfirmation,
+  end;
+}
+  if Pref.ConfirmOnExit then
+    if MessageDlg('Yasagure Client\82ð\8fI\97¹\82µ\82Ü\82·', mtConfirmation,
                 mbOkCancel, 0) = mrCancel then
+    begin
+      CanClose := false;
+      exit;
+    end;
+
+  //\83t\83@\83C\83\8b\82Ì\8dX\90V\83`\83F\83b\83N\81i\83G\83f\83B\83^\95\94\81j
+  if CheckFileModified(Self) = idCancel then
+  begin
     CanClose := false;
+    exit;
+  end;
+  //\83\8d\83O\81i\83^\83u\81j\93à\97e\82Ì\95Ï\8dX\83`\83F\83b\83N
+  if frmLog.CheckLog(Self) = idCancel then
+  begin
+    CanClose := false;
+    exit;
+  end;
+  DeleteBackupFile(FScriptBackupFile);   // \88ê\8e\9e\83t\83@\83C\83\8b\8dí\8f\9c\82Ö
 end;
 
 procedure TfrmSender.UpdateIfGhostBox;
@@ -2310,7 +2432,7 @@ begin
   SysUtils.Beep;
   Beep;
   ShowHintLabel('SSTP Bottle\83T\81[\83o\82Æ\82Ì\90Ú\91±\82É\8e¸\94s\82µ\82Ü\82µ\82½', WarningColor);
-  ShowMessage('SSTP Bottle\83T\81[\83o\82Æ\82Ì\90Ú\91±\82É\8e¸\94s\82µ\82Ü\82µ\82½'#13#10 +
+  frmMessageBox.ShowMessage('SSTP Bottle\83T\81[\83o\82Æ\82Ì\90Ú\91±\82É\8e¸\94s\82µ\82Ü\82µ\82½'#13#10 +
     (Sender as THTTPDownloadThread).LastErrorMessage);
   Connecting := false;
 end;
@@ -2384,8 +2506,11 @@ procedure TfrmSender.RetryBeginConnect;
 begin
   if FBeginConnectFailCount < 3 then begin
     // \90Ø\92f\82³\82ê\82Ä\82¢\82ê\82Î\8dÄ\90Ú\91±
+    IdSlpp20.OnDisconnect := nil;
+    if IdSlpp20.Connected then IdSlpp20.Disconnect;
     FAutoAddAfterGetChannel := true;
     BeginConnect;
+    IdSlpp20.OnDisconnect := Slpp20Disconnect;
   end else if FBeginConnectFailCount = 3 then begin
     frmLog.AddCurrentSystemLog('SYSTEM', '\8dÄ\90Ú\91±\8e©\93®\83\8a\83g\83\89\83C\82ð\92\86\8e~\82µ\82Ü\82·');
     frmMessageBox.ShowMessage(
@@ -2633,9 +2758,9 @@ begin
   frmLog.SetBottleState(MID, lsPlaying);
 end;
 
-procedure TfrmSender.actInsertCueExecute(Sender: TObject);
+procedure TfrmSender.LogInsertCue(TestAction, SelectAll: boolean);
 var InsertItem: TLogItem;
-    i, errCount, Res: integer;
+    i, errCount, Res, Tstart: integer;
     Log: TBottleLogList;
     ErrorMes: String; // \83X\83N\83\8a\83v\83g\82Ì\83G\83\89\81[\82Ì\93à\97e
 begin
@@ -2652,50 +2777,40 @@ begin
   if Log = nil then Exit;
   FBottleSSTP.OnResendCountChange := nil;
   errCount := 0;
-  for i := frmLog.lvwLog.Selected.Index downto 0 do begin
+  if SelectAll then Tstart := frmLog.lvwLog.Items.Count-1  // \8ew\92è\82³\82ê\82½\83^\83u\82Ì\91S\90\94
+  else Tstart := frmLog.lvwLog.Selected.Index;             // \91I\91ð\82³\82ê\82Ä\82¢\82é\8fê\8f\8a\82©\82ç
+  for i := Tstart downto 0 do begin
     if (Log[i] as TLogItem).LogType <> ltBottle then Continue;
     InsertItem := TLogItem.Create(Log[i] as TLogItem);
     try
       //\83A\83N\83V\83\87\83\93\82ª\97L\8cø\82Å\82 \82ê\82Î\83X\83N\83\8a\83v\83g\82Ì\95Ï\8a·\82µ\82È\82¢
-      if Pref.LogAction then
-        ErrorMes := ''
-      else
-        InsertItem.Script := ScriptTransForSSTP(InsertItem.Script, ErrorMes);
-
-      if ErrorMes <> '' then
+      if TestAction then
       begin
-        Res := MessageDlg('\83X\83N\83\8a\83v\83g\82É\96â\91è\82ª\82 \82é\89Â\94\\90«\82ª\82 \82è\82Ü\82·\81B' +
-          '\8dÄ\90\82µ\82Ü\82·\82©?'#13#10 + ErrorMes, mtWarning,
-          mbYesNoCancel, 0);
-        if Res = mrNo then
-          raise Exception.Create('Script Syntax Error')
-        else if Res = mrCancel then
-        begin
-          InsertItem.Free;
-          FBottleSstp.Clear;
-          frmLog.AllBottleOpened;
-          Break;
-        end;
-      end;
-      //\83A\83N\83V\83\87\83\93\82ª\97L\8cø\82Å\82 \82ê\82Î\92Ê\8fí\8f\88\97\9d\8fÈ\97ª
-      if Pref.LogAction then
-      begin
-        //\8c^\95Ï\8a·\82Æ\8eó\90M
-        BottleCnv(InsertItem.Mid
-                 ,InsertItem.Channel
-                 ,InsertItem.Ghost
-                 ,InsertItem.Script
-                 ,InsertItem.Votes
-                 ,InsertItem.Agrees
-                 ,InsertItem.LogTime);
+        BottleCnv(InsertItem); // \8c^\95Ï\8a·\82Æ\8eó\90M
+        frmLog.StatusBar.Panels[1].Text := Format('\8ec\82è%d\8c\8f', [i]);
+        frmLog.StatusBar.Update;
       end else
       begin
+        InsertItem.Script := ScriptTransForSSTP(InsertItem.Script, ErrorMes);
+        if ErrorMes <> '' then
+        begin
+          Res := MessageDlg('\83X\83N\83\8a\83v\83g\82É\96â\91è\82ª\82 \82é\89Â\94\\90«\82ª\82 \82è\82Ü\82·\81B' +
+            '\8dÄ\90\82µ\82Ü\82·\82©?'#13#10 + ErrorMes, mtWarning,
+            mbYesNoCancel, 0);
+          if Res = mrNo then
+            raise Exception.Create('Script Syntax Error')
+          else if Res = mrCancel then
+          begin
+            InsertItem.Free;
+            FBottleSstp.Clear;
+            frmLog.AllBottleOpened;
+            Break;
+          end;
+        end;
         //\83`\83\83\83\93\83l\83\8b\83S\81[\83X\83g\91Î\8dô
         if InsertItem.Ghost = '' then
-        begin
           if ChannelList.Channel[InsertItem.Channel] <> nil then
             InsertItem.Ghost := ChannelList.Channel[InsertItem.Channel].Ghost;
-        end;
         FBottleSSTP.Push(InsertItem);
         frmLog.SetBottleState(InsertItem.MID, lsUnopened);
       end;
@@ -2704,6 +2819,8 @@ begin
       Inc(errCount);
     end;
   end;
+  if TestAction then
+    frmLog.StatusBar.Panels[1].Text := '\8a®\97¹';
   if errCount > 0 then
     ShowMessage(Format('%d\8c\8f\82Ì\83{\83g\83\8b\82É\96â\91è\82ª\82 \82Á\82½\82½\82ß\8dÄ\90\82Å\82«\82Ü\82¹\82ñ\81B', [errCount]));
   FBottleSSTP.OnResendCountChange := BottleSstpResendCountChange;
@@ -2711,6 +2828,11 @@ begin
   frmLog.lvwLog.Invalidate;
 end;
 
+procedure TfrmSender.actInsertCueExecute(Sender: TObject);
+begin
+  LogInsertCue(false, false); // \92Ê\8fí\82Ì\98A\91±\8dÄ\90\82ð\82·\82é
+end;
+
 function TfrmSender.ScriptTransForSSTP(const Script: String;
   out Error: String): String;
 var TransOpt: TScriptTransOptions;
@@ -2816,7 +2938,7 @@ begin
     ghost := cbxTargetGhost.Text
   else
     ghost := '';
-{
+{  \8eè\93®\82Ì\82Í\82¸
   else if FNowChannel <> '' then
     ghost := ChannelList.Channel[FNowChannel].Ghost;
 }
@@ -2890,7 +3012,7 @@ begin
   if FScriptBuffer.Count = 0 then
     Exit;
   memScript.Lines.Assign(FScriptBuffer[0] as TStringList);
-  memScriptChange(Self);
+  EditerStatusChange;
   ShowHintLabel('\83X\83N\83\8a\83v\83g\82ð\8cÄ\82Ñ\8fo\82µ\82Ü\82µ\82½');
 end;
 
@@ -2904,7 +3026,7 @@ begin
       Ghost := '';
       if cbxTargetGhost.ItemIndex > 0 then
         Ghost := cbxTargetGhost.Text;
-{
+{ \8eè\93®\82Å\8ew\92è\82³\82ê\82é
       else if FNowChannel <> '' then
         Ghost := ChannelList.Channel[FNowChannel].Ghost;
 }
@@ -3066,6 +3188,7 @@ begin
   else
     Ghost := '';
   frmLog.AddCurrentScriptLog('\83N\83\8a\83b\83v', Script, ClipChannel, '', Ghost, Now(), 0, 0);
+  SetFileModified(false); // \95Ï\8dX\82³\82ê\82Ä\82¢\82È\82¢\82±\82Æ\82É\82µ\82Ä\83N\83\8a\83A\82³\82¹\82é
   ClearEditor;
 end;
 
@@ -3093,17 +3216,25 @@ begin
 end;
 
 procedure TfrmSender.CopyFromLogToEditor(Log: TLogItem);
+var
+  Ghost: string;
 begin
   if Log.LogType <> ltBottle then Exit;
   frmSender.actClear.Execute; // \8c»\8dÝ\82Ì\83X\83N\83\8a\83v\83g\82ð\83N\83\8a\83b\83v\82·\82é(\90Ý\92è\82É\82æ\82Á\82Ä)
   memScript.Lines.Clear;
   memScript.Lines.Add(Log.Script);
-  if Log.Ghost <> '' then
+  SetFileName(Log.MID, true, false);  // \81ª\82Å\93à\97e\82ª\8dX\90V\82³\82ê\82é\82Ì\82Å\8dÄ\93x\90V\8bK\82É
+  Ghost := Log.Ghost;
+  //\83`\83\83\83\93\83l\83\8b\83S\81[\83X\83g\91Î\8dô
+  if Ghost = '' then
+    if ChannelList.Channel[Log.Channel] <> nil then
+      Ghost := ChannelList.Channel[Log.Channel].Ghost;
+  if Ghost <> '' then
   begin
     // \83S\81[\83X\83g\96¼\82ð\83{\83b\83N\83X\82É\93ü\82ê\82é
     // \96³\97\9d\96î\97\9d\83S\81[\83X\83g\96¼\82ð\92Ç\89Á\82µ\82Ä\82©\82ç\8dÄ\8d\\92z\82·\82é\82±\82Æ\82Å
     // \96³\97\9d\96î\97\9d\83S\81[\83X\83g\96¼\82ª\83{\83b\83N\83X\82É\93ü\82é
-    cbxTargetGhost.Items.Add(Log.Ghost);
+    cbxTargetGhost.Items.Add(Ghost);
     cbxTargetGhost.ItemIndex := cbxTargetGhost.Items.Count-1;
     UpdateIfGhostBox;
     cbxTargetGhost.Invalidate;
@@ -3122,6 +3253,7 @@ begin
   frmLog.SelectedBottleLog.Delete(frmLog.lvwLog.Selected.Index);
   frmLog.UpdateWindow;
   frmLog.lvwLogChange(Self, nil, ctState);
+  frmLog.SelectedBottleLog.LogModified := true; // \95Ï\8dX\82³\82ê\82½
 end;
 
 procedure TfrmSender.ClearEditor;
@@ -3130,6 +3262,10 @@ var TmpScript: String;
     DoSaveBuffer: boolean;
     SavedScript: TStringList;
 begin
+  // \90Ý\92è\82É\82æ\82è\83N\83\8a\83A\8e\9e\82É\83t\83@\83C\83\8b\83Z\81[\83u\82·\82é\82©\8am\94F\82·\82é
+  // \8e©\93®\83N\83\8a\83b\83v\82ª\97L\8cø\82Å\82 \82ê\82Î\83X\83L\83b\83v\82·\82é
+  if (NOT Pref.NotCheckClear) AND (NOT Pref.AutoClip) then
+    if CheckFileModified(Self) = idCancel then abort;
   // \83X\83N\83\8a\83v\83g\82Ì\83N\83\8a\83A
   // \82Ü\82¸\81A\83X\83N\83\8a\83v\83g\83N\83\8a\83A\83o\83b\83t\83@\82É\8c»\8dÝ\82Ì\83X\83N\83\8a\83v\83g\82ð\95Û\91\82·\82é
   DoSaveBuffer := false;
@@ -3158,11 +3294,13 @@ begin
   memScript.SelStart := Position-1;
 
   if Visible then memScript.SetFocus;
-  FScriptModified := false;
-  memScriptChange(self);
+//  FScriptModified := false;
+
+  SetFileName(NewFileTitle, true, true);    // \83t\83@\83C\83\8b\82ð\90V\8bK\88µ\82¢\82É\82·\82é
+  EditerStatusChange;                       // \83X\83e\81[\83^\83X\8dX\90V
 end;
 
-procedure TfrmSender.AppendTextLog(const FileName, Line: String);
+procedure TfrmSender.AppendTextLog(const FileName, Line: String; FAppend: boolean);
 var
   F: TextFile;
 begin
@@ -3170,7 +3308,7 @@ begin
   try
     ForceDirectories(ExtractFileDir(FileName));
     AssignFile(F, FileName);
-    if FileExists(FileName) then
+    if FileExists(FileName) AND FAppend then  // \92Ç\8bL\82©\8fã\8f\91\82«\94»\92è
       Append(F)
     else
       Rewrite(F);
@@ -3276,7 +3414,7 @@ begin
     StatusBar.Panels[PanelBytes].Text := Format('(%d\83o\83C\83g)', [Length(SelText)]);
   end else
   begin
-    memScriptChange(Self);
+    EditerStatusChange;
   end;
 end;
 
@@ -3345,12 +3483,14 @@ begin
   end;
 end;
 
+{
 procedure TfrmSender.WMQueryEndSession(var msg: TWMQueryEndSession);
 begin
   // Windows\82ª\8fI\97¹\82µ\82æ\82¤\82Æ\82µ\82Ä\82¢\82é\82Ì\82ð\8a´\92m\82·\82é
   FEndSession := true;
   inherited;
 end;
+}
 
 {
 procedure TfrmSender.IdSLPP20ConnectFailed(Sender: TObject);
@@ -3360,11 +3500,11 @@ begin
   begin
     Beep;
     if Pref.UseHttpProxy then
-      ShowMessage('HTTP Proxy\82ð\92Ê\82\82ÄSSTP Bottle\83T\81[\83o\82É\90Ú\91±\82Å\82«\82Ü\82¹\82ñ\82Å\82µ\82½\81B'#13#10 +
+      frmMessageBox.ShowMessage('HTTP Proxy\82ð\92Ê\82\82ÄSSTP Bottle\83T\81[\83o\82É\90Ú\91±\82Å\82«\82Ü\82¹\82ñ\82Å\82µ\82½\81B'#13#10 +
                   '\83l\83b\83g\83\8f\81[\83N\82Ì\8fó\91Ô\81EProxy\82Ì\8fó\91Ô\82ð\8am\94F\82µ\82Ä\82­\82¾\82³\82¢\81B'#13#10 +
                   '\82 \82é\82¢\82Í\83T\81[\83o\82ª\83_\83E\83\93\82µ\82Ä\82¢\82é\89Â\94\\90«\82ª\82 \82è\82Ü\82·\81B')
     else
-      ShowMessage('SSTP Bottle\83T\81[\83o\82É\90Ú\91±\82Å\82«\82Ü\82¹\82ñ\82Å\82µ\82½\81B'#13#10 +
+      frmMessageBox.ShowMessage('SSTP Bottle\83T\81[\83o\82É\90Ú\91±\82Å\82«\82Ü\82¹\82ñ\82Å\82µ\82½\81B'#13#10 +
                   '\83l\83b\83g\83\8f\81[\83N\82É\8cq\82ª\82Á\82Ä\82¢\82é\82©\8am\94F\82µ\82Ä\82­\82¾\82³\82¢\81B'#13#10 +
                   '\82 \82é\82¢\82Í\83T\81[\83o\82ª\83_\83E\83\93\82µ\82Ä\82¢\82é\89Â\94\\90«\82ª\82 \82è\82Ü\82·\81B');
   end;
@@ -3373,47 +3513,94 @@ begin
 end;
 }
 
-procedure TfrmSender.BottleCnv(Mid, Channel, Ghost, Script: String;
-  Vote, Agree: integer; LogTime: TDateTime);
+procedure TfrmSender.BuildReplaceMenu(Root: TMenuItem);
+var
+  i: integer;
+  Presets: TReplacePresetCollection;
+  NewItem: TMenuItem;
+begin
+  Root.Clear;
+  Presets := Pref.ReplacePresets.Presets;
+  for i := 0 to Presets.Count-1 do
+  begin
+    NewItem := TMenuItem.Create(Root);
+    with NewItem do
+    begin
+      Caption := Presets[i].Title;
+      ShortCut := Presets[i].ShortCut;
+      AutoHotkeys := maManual;
+      Hint := Presets[i].Pairs.StringExpression;
+      Tag := i;
+      OnClick := mnPresetReplaceClick;
+    end;
+    Root.Add(NewItem);
+  end;
+  Root.Enabled := Presets.Count > 0;
+end;
+
+procedure TfrmSender.mnPresetReplaceClick(Sender: TObject);
+var
+  Preset: TReplacePreset;
+  Lines, New: string;
+begin
+  // \83v\83\8a\83Z\83b\83g\92u\8a·\82ð\8eÀ\8ds
+  Preset := Pref.ReplacePresets.Presets[(Sender as TMenuItem).Tag];
+  Lines := memScript.Lines.Text;
+  New := Preset.Pairs.ExecuteReplace(Lines);
+  if New <> Lines then
+  begin
+    memScript.SelectAll;
+    memScript.SelText := New;
+  end;
+  if Preset.ConfirmAfterReplace then
+  begin
+    btnConfirm.Click;
+  end;
+end;
+
+procedure TfrmSender.BottleCnv(Log: TLogItem);
 var
   HeadValue: THeadValue;
 begin
-  //\83T\81[\83o\82©\82ç\82Ì\83\81\83b\83Z\81[\83W\82ð\89¼\91z\92è\8b`
+  //HeadValue\8c^\82Ö
   HeadValue := THeadValue.Create;
-  HeadValue['MID'] := MID;
-  HeadValue['Channel'] := Channel;
-  HeadValue['SEND'] := 'SSTP/1.4';
-  HeadValue['Sender'] := 'SSTP Bottle ' + Channel;
-  HeadValue['IfGhost'] := Ghost;
-  HeadValue['Script'] := Script;
-  HeadValue['Charset'] := 'Shift_JIS';
+  HeadValue['MID'] := Log.MID;
+  HeadValue['Channel'] := Log.Channel;
+//  HeadValue['SEND'] := 'SSTP/1.4';
+//  HeadValue['Sender'] := 'SSTP Bottle ' + Log.Channel;
+  HeadValue['IfGhost'] := Log.Ghost;
+  HeadValue['Script'] := Log.Script;
+//  HeadValue['Charset'] := 'Shift_JIS';
+//  HeadValue['DialogMessage'] := Log.Script;
   try
     //\83\81\83b\83Z\81[\83W\8eó\90M\8f\88\97\9d\82Ö
-    DispatchBottle(HeadValue, LogTime, Vote, Agree);
+    DispatchBottle(HeadValue, Log.LogTime, Log.Votes, Log.Agrees);
   finally
     HeadValue.Free;
   end;
 end;
 
-procedure TfrmSender.SetScriptAutoBackUp;
+procedure TfrmSender.SetScriptAutoBackup;
 begin
   //\83o\83b\83N\83A\83b\83v\83^\83C\83}\81[\82Ö\92l\82Ì\83Z\83b\83g
-  frmSender.ScriptBackUp.Enabled := Pref.ScriptBackUp;
-  frmSender.ScriptBackUp.Interval := Pref.ScriptBackUpTime * 60000;
+  frmSender.ScriptBackup.Enabled := Pref.ScriptBackup;
+  frmSender.ScriptBackup.Interval := Pref.ScriptBackupTime * 60000;
 end;
 
-procedure TfrmSender.ScriptBackUpTimer(Sender: TObject);
+procedure TfrmSender.ScriptBackupTimer(Sender: TObject);
 var
   Script: string;
 begin
-  //\83X\83N\83\8a\83v\83g\82Ì\8e©\93®\83o\83b\83N\83A\83b\83v
-  if FScriptBackModified then
+  // \83X\83N\83\8a\83v\83g\82Ì\83o\83b\83N\83A\83b\83v
+  if FScriptBackupModified then
   begin
-    Script := GetScriptText;
-    AppendTextLog(ExtractFilePath(Application.ExeName) + 'Script.bak',
-      FormatDateTime('yy/mm/dd hh:nn:ss', Now) + #13#10 + Script);
+    Script := FormatDateTime('yy/mm/dd hh:nn:ss', Now) + #13#10 + GetScriptText;
+
+    AppendTextLog(FScriptBackupDir + FScriptBackupFile, Script,
+      not Pref.ScriptBackupMode);
+
     ShowHintLabel('\83o\83b\83N\83A\83b\83v\8eÀ\8ds');
-    FScriptBackModified := False;
+    FScriptBackupModified := False;
   end;
 end;
 
@@ -3422,10 +3609,16 @@ var
   Ch: TChannelListItem;
   i: integer;
 begin
-  //\83`\83\83\83\93\83l\83\8b\83S\81[\83X\83g\92è\8b`
+  // \83`\83\83\83\93\83l\83\8b\83S\81[\83X\83g\92è\8b`
   ChannelList.Clear;
-  //\83t\83\89\83O\82ð\8c©\82Ä\94»\92f
+  // \83t\83\89\83O\82ð\8c©\82Ä\94»\92f
   if NOT Pref.GhostCh then exit;
+  if (Pref.GhostChList.Count mod 2) = 1 then
+  begin
+    Pref.GhostChList.Add(''); // \95s\91«\82µ\82Ä\82¢\82é\82Ì\82Å\93K\93\96\82É\92Ç\89Á
+    ShowMessage('\83`\83\83\83\93\83l\83\8b\82ð\8dÄ\90Ý\92è\82µ\82Ä\82­\82¾\82³\82¢');
+    exit;
+  end;
   i := 0;
   while i < Pref.GhostChList.Count-1 do
   begin
@@ -3443,8 +3636,9 @@ end;
 
 procedure TfrmSender.UpdateJoinChannelList;
 begin
-  UpdateChannelList;  //\83`\83\83\83\93\83l\83\8b\90Ý\92è
-  with tabChannel do begin
+  UpdateChannelList;  // \83`\83\83\83\93\83l\83\8b\90Ý\92è
+  with tabChannel do  // \96¢\8eg\97p
+  begin
     OnChange := nil;
     Tabs.BeginUpdate;
     Tabs.Clear;
@@ -3456,45 +3650,201 @@ begin
   end;
 end;
 
+procedure TfrmSender.PerformFileOpen(const AFileName: string);
+begin
+  // \8eÀ\8dÛ\82É\83t\83@\83C\83\8b\82ð\8aJ\82­
+  Screen.Cursor := crHourGlass;
+  try
+    memScript.Lines.LoadFromFile(AFileName);
+    SetFileName(AFileName, true, true);
+    Pref.TextDir := ExtractFileDir(AFileName);
+  except
+    on E: Exception do
+    begin
+      ShowMessage('\83G\83\89\81[:'#13#10#13#10 + E.Message);
+      SetFileName(NewFileTitle, true, true);  // \8e¸\94s\82µ\82½\8e\9e\82Í\83^\83C\83g\83\8b\8f\89\8aú\89»\81B
+    end;
+  end;
+  Screen.Cursor := crDefault;
+  memScript.SetFocus;
+  EditerStatusChange;
+end;
 
-procedure TfrmSender.BuildReplaceMenu(Root: TMenuItem);
+procedure TfrmSender.PerformFileSave(const AFileName: string);
+begin
+  // \8eÀ\8dÛ\82É\95Û\91\82·\82é
+  Screen.Cursor := crHourGlass;
+  try
+    memScript.Lines.SaveToFile(AFileName);
+    SetFileModified(false);
+    Pref.TextDir := ExtractFileDir(AFileName);
+  except
+    on E: Exception do
+      ShowMessage('\83G\83\89\81[:'#13#10#13#10 + E.Message);
+  end;
+  Screen.Cursor := crDefault;
+end;
+
+procedure TfrmSender.actFileOpenExecute(Sender: TObject);
+begin
+  // \8aJ\82­
+  if CheckFileModified(Self) = idCancel then exit;
+  OpenDialog.InitialDir := Pref.TextDir;
+  if OpenDialog.Execute then
+    if CheckFileExt(OpenDialog.FileName) then // \8ag\92£\8eq\82Ì\83`\83F\83b\83N
+      PerformFileOpen(OpenDialog.FileName);
+end;
+
+procedure TfrmSender.SetFileName(const FileName: string; Reset, SetNewFile: boolean);
 var
-  i: integer;
-  Presets: TReplacePresetCollection;
-  NewItem: TMenuItem;
+  FPath: string;
 begin
-  Root.Clear;
-  Presets := Pref.ReplacePresets.Presets;
-  for i := 0 to Presets.Count-1 do
+  // \83^\83C\83g\83\8b\95Ï\8dX\82Æ\83t\83@\83C\83\8b\83p\83X\83Z\83b\83g
+  // \82à\82µ\81A\91æ\93ñ\88ø\90\94\82ªfalse\82È\82ç\83^\83C\83g\83\8b\82¾\82¯\95Ï\8dX
+  if Reset then
   begin
-    NewItem := TMenuItem.Create(Root);
-    with NewItem do
+    SetFileModified(false);               // \95Ï\8dX\96³\82µ\82É
+    FScriptBackupModified := false;       // \8e©\93®\95Û\91\88ê\8e\9e\92â\8e~
+    DeleteBackupFile(FScriptBackupFile);  // \88ê\8e\9e\83t\83@\83C\83\8b\8dí\8f\9c\82Ö
+  end;
+
+  if SetNewFile then
+    FFileName := FileName;                // \8d¡\8c»\8dÝ\82Ì\95Ò\8fW\83t\83@\83C\83\8b\96¼
+
+  FPath := ExtractFileName(FileName);
+
+  if Pref.ScriptBackupMode then
+    FScriptBackupFile := FormatDateTime('yymmddhhnnss', Now) + FPath
+  else
+    FScriptBackupFile := StdBackupFile;
+  Self.Caption := Format('%s - %s', [FPath, FOriginalCaption]);
+end;
+
+function TfrmSender.CheckFileModified(Sender: TObject): integer;
+var
+  Res: integer;
+begin
+  // \83t\83@\83C\83\8b\82Ì\95Ï\8dX\82ð\8am\94F
+  // Result := idCancel\82Å\8fI\97¹\91j\8e~
+  if FFileModified then
+  begin
+    Res := MessageDlg(Format('\83t\83@\83C\83\8b %s \82Í\95Ï\8dX\82³\82ê\82Ä\82¢\82Ü\82·\81B'#13#10#13#10 +
+      '\95Û\91\82µ\82Ü\82·\82©\81H', [FFileName]), mtConfirmation, mbYesNoCancel, 0);
+    if Res = idYes then Res := FileSave(Self);
+  end else
+    Res := idNo;
+  Result := Res;
+end;
+
+function TfrmSender.FileSave(Sender: TObject): integer;
+var
+  Res: integer;
+begin
+  // \8fã\8f\91\82«
+  if FFileName = NewFileTitle then
+    Res := FileSaveAs(Sender)       // \96¼\91O\82ð\95t\82¯\82Ä\95Û\91\82Ö
+  else
+  begin
+    PerformFileSave(FFileName);     // \8eÀ\8dÛ\82É\95Û\91\82·\82é\82Ö
+    Res := idNo;
+  end;
+  Result := Res;
+end;
+
+function TfrmSender.FileSaveAs(Sender: TObject): integer;
+var
+  Res: integer;
+begin
+  // \96¼\91O\82ð\95t\82¯\82Ä\95Û\91
+  Res := idYes;
+  SaveDialog.FileName := ExtractFileName(FFileName);
+  SaveDialog.InitialDir := Pref.TextDir;
+  if SaveDialog.Execute then
+  begin
+    if CheckFileExt(SaveDialog.FileName) = false then // \8ag\92£\8eq\83`\83F\83b\83N
     begin
-      Caption := Presets[i].Title;
-      ShortCut := Presets[i].ShortCut;
-      AutoHotkeys := maManual;
-      Hint := Presets[i].Pairs.StringExpression;
-      Tag := i;
-      OnClick := mnPresetReplaceClick;
+      Res := idCancel;
+      abort;
     end;
-    Root.Add(NewItem);
+    if FileExists(SaveDialog.FileName) then
+      Res := MessageDlg(Format('%s \82Í\8aù\82É\91\8dÝ\82µ\82Ä\82¢\82Ü\82·\81B'#13#10 + '\8fã\8f\91\82«\82µ\82Ü\82·\82©\81H',
+        [SaveDialog.FileName]), mtConfirmation, mbYesNoCancel, 0);
+    if Res = idYes then
+    begin
+      PerformFileSave(SaveDialog.FileName);
+      SetFileName(SaveDialog.FileName, true, true);
+    end;
+  end else
+    Res := idCancel;
+  Result := Res;
+end;
+
+procedure TfrmSender.actFileSaveAsExecute(Sender: TObject);
+begin
+  // \96¼\91O\82ð\95t\82¯\82Ä\95Û\91\8cÄ\82Ñ\8fo\82µ
+  FileSaveAs(Self);
+end;
+
+procedure TfrmSender.actFileSaveExecute(Sender: TObject);
+begin
+  // \8fã\8f\91\82«\95Û\91\8cÄ\82Ñ\8fo\82µ
+  FileSave(Self);
+end;
+
+procedure TfrmSender.SetFileModified(Value: boolean);
+begin
+  // \83t\83@\83C\83\8b\95Ï\8dX\83t\83\89\83O\82Æ\95\\8e¦\95Ï\8dX
+  if Value then
+    StatusBar.Panels[PanelConnecting].Text := '\95Ï\8dX'
+  else
+    StatusBar.Panels[PanelConnecting].Text := '';
+  FFileModified := Value;
+end;
+
+procedure TfrmSender.OpenBrowser(const Url: string);
+var
+  Command: string;
+begin
+  if Pref.BrowserExeName = '' then
+  begin
+    ShellExecute(Handle, 'open', PChar(Url), nil, nil, SW_SHOW);
+  end else
+  begin
+    Command := Pref.BrowserExeName + ' ' + Url;
+    WinExec(PChar(Command), SW_SHOW);
   end;
-  Root.Enabled := Presets.Count > 0;
 end;
 
-procedure TfrmSender.mnPresetReplaceClick(Sender: TObject);
+function TfrmSender.CheckFileExt(const FileName: string): boolean;
 var
-  Preset: TReplacePreset;
-  Lines, New: string;
+  Res: boolean;
 begin
-  // \83v\83\8a\83Z\83b\83g\92u\8a·\82ð\8eÀ\8ds
-  Preset := Pref.ReplacePresets.Presets[(Sender as TMenuItem).Tag];
-  Lines := memScript.Lines.Text;
-  New := Preset.Pairs.ExecuteReplace(Lines);
-  if New <> Lines then
+  // \8ag\92£\8eq.bak\82Ì\83t\83@\83C\83\8b\82Í\93Ç\8f\91\82«\82µ\82È\82¢\81B
+  Res := true;
+  if LowerCase(ExtractFileExt(FileName)) = '.bak' then  // StdBackupFile\82Æ\93¯\8ag\92£\8eq
   begin
-    memScript.SelectAll;
-    memScript.SelText := New;
+    beep;
+    ShowMessage('\8ag\92£\8eqbak\82Í\88µ\82¦\82Ü\82¹\82ñ\81B'#13#10 + '\91¼\82Ì\83G\83f\83B\83^\82ð\8eg\97p\82µ\82Ä\82­\82¾\82³\82¢\81B');
+    Res := false;
+  end;
+  Result := Res;
+end;
+
+procedure TfrmSender.DeleteBackupFile(const FileName: string);
+var
+  FPath: string;
+begin
+  // \8fã\8f\91\95Û\91\82Ì\88ê\8e\9e\83t\83@\83C\83\8b\82ð\8dí\8f\9c
+  if Pref.NoDeleteBackupFile or (FileName = '') or
+    (StdBackupFile = FileName) then exit;
+
+  FPath := FScriptBackupDir + FileName;
+  try
+    if FileExists(FPath) then
+      DeleteFile(FPath);
+  except
+    on E: Exception do
+      ShowMessage('\88ê\8e\9e\83t\83@\83C\83\8b\8dí\8f\9c\83G\83\89\81[:'#13#10#13#10 + E.Message);
   end;
 end;