dexed/etc/terminal/terminalctrls.pas

674 lines
18 KiB
Plaintext

unit TerminalCtrls;
{$mode delphi}
{$ifdef lclgtk2}
{$ifdef unix}
{$define hasgtk2term}
{$endif}
{$endif}
interface
uses
Gtk2Term, Classes, SysUtils, Controls, math,
{$ifdef windows} windows, AsyncProcess,{$endif}
Graphics, dialogs;
type
TTerminalScrollInfo = record
min, max, value, pageSize: integer;
end;
TTerminalTextScrolled = procedure(sender: TObject; delta: integer) of Object;
TASCIIControlCharacter = (
HOME = 1,
LEFT = 2,
ETX = 3, // interrupt, clear temp line
SUPR = 4,
&END = 5,
RIGHT = 6,
BS = 8,
LF = 10,
VT = 11, // delete from cursor pos to end of line
CR = 13
);
TTerminal = class(TCustomControl)
private
FInfo: Pointer;
{$ifdef hasgtk2term}
fTerminalHanlde: PVteTerminal;
{$endif}
{$ifdef windows}
fTermProgram: string;
fTermProcess: TAsyncProcess;
fTermWnd: HWND;
{$endif}
fOnTerminate: TNotifyEvent;
fOnTerminalVisibleChanged: TNotifyEvent;
fOnTerminalTextScrolled: TTerminalTextScrolled;
fBackgroundColor: TColor;
fForegroundColor: TColor;
fSelectedColor: TColor;
fScrollbackLines: LongWord;
fScrollOnOutput: boolean;
fScrollOnKeyStroke: boolean;
procedure setBackgroundColor(value: TColor);
procedure setForegroundColor(value: TColor);
procedure setSelectedColor(value: TColor);
procedure setScrollBackLines(value: LongWord);
procedure setScrollOnOutput(value: boolean);
procedure setScrollOnKeyStroke(value: boolean);
protected
// Only used at design-time.
procedure Paint; override;
procedure DoTerminate; virtual;
procedure DoTerminalVisibleChanged; virtual;
procedure DoTerminalTextScrolled(delta: integer); virtual;
procedure FontChanged(Sender: TObject); override;
public
constructor Create(AOwner: TComponent); override;
destructor destroy; override;
procedure Restart;
procedure Reparent;
// Sends a command, as it would be manually typed. Line feed is automatically added.
procedure Command(const data: string);
procedure SendControlChar(const cc: TASCIIControlCharacter);
procedure copyToClipboard();
procedure pasteFromClipboard();
function getLine(const line: integer): string;
function getWholeText(): string;
function getCursorPosition: TPoint;
function getVScrollInfo: TTerminalScrollInfo;
procedure setVScrollPosition(i: integer);
published
{$ifdef windows}
property terminalProgram: string read fTermProgram write fTermProgram;
{$endif}
// set if view is scrolled to the back on key stroke
property scrollOnOutut: boolean read fScrollOnKeyStroke write setScrollOnKeyStroke default true;
// set if view is scrolled to the back on output
property scrollOnKeyStroke: boolean read FScrollOnOutput write setScrollOnOutput default true;
// set the back buffer in line count
property scrollbackLines: LongWord read fScrollbackLines write setScrollBackLines default 4096;
// Background color
property backgroundColor: TColor read fBackgroundColor write setBackgroundColor default clBlack;
// Font color
property foregroundColor: TColor read fForegroundColor write setForegroundColor default clWhite;
// Background color for the selection
property selectedColor: TColor read fSelectedColor write setSelectedColor default clWhite;
property Align;
property Anchors;
property Constraints;
property DockSite;
property DragCursor;
property DragKind;
property DragMode;
property DoubleBuffered;
property Enabled;
// The name and height properties are handled. see foregroundColor for Color.
property Font;
property ParentShowHint;
property PopupMenu;
property ShowHint;
property TabOrder;
property TabStop;
property UseDockManager default True;
property Visible;
property OnClick;
property OnContextPopup;
property OnDockDrop;
property OnDockOver;
property OnDblClick;
property OnDragDrop;
property OnDragOver;
property OnEndDock;
property OnEndDrag;
property OnEnter;
property OnExit;
property OnGetSiteInfo;
property OnGetDockCaption;
property OnMouseDown;
property OnMouseEnter;
property OnMouseLeave;
property OnMouseMove;
property OnMouseUp;
property OnMouseWheel;
property OnMouseWheelDown;
property OnMouseWheelUp;
property OnResize;
property OnStartDock;
property OnStartDrag;
property OnUnDock;
property OnTerminate: TNotifyEvent read FOnTerminate write FOnTerminate;
// Note: The hosted widget is there and visual settings can be applied.
// In many cases DoFirstShow, OnShow and likes will happen too quickly.
property OnTerminalVisibleChanged: TNotifyEvent read fOnTerminalVisibleChanged write fOnTerminalVisibleChanged;
property OnTextScrolled: TTerminalTextScrolled read fOnTerminalTextScrolled write fOnTerminalTextScrolled;
end;
function TerminalAvailable: Boolean;
implementation
{$ifdef hasgtk2term}
uses
LCLType, WSControls, WSLCLClasses, GLib2, Gtk2, Gtk2Def, Gtk2Proc,
Gtk2WSControls, gdk2;
type
TGtk2WSTerminal = class(TWSCustomControl)
private
class procedure SetCallbacks(const AGtkWidget: PGtkWidget;
const AWidgetInfo: PWidgetInfo); virtual;
published
class function CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): TLCLIntfHandle; override;
end;
procedure TerminalCommit(Widget: PGtkWidget; c: gchar; s: guint; user: Pointer); cdecl;
var
Info: PWidgetInfo;
begin
Info := PWidgetInfo(g_object_get_data(PGObject(Widget), 'widgetinfo'));
TTerminal(Info.LCLObject).DoTerminalTextScrolled(1);
end;
procedure TerminalExit(Widget: PGtkWidget; status: gint; user: Pointer); cdecl;
var
Info: PWidgetInfo;
begin
Info := PWidgetInfo(g_object_get_data(PGObject(Widget), 'widgetinfo'));
TTerminal(Info.LCLObject).DoTerminate;
end;
procedure TerminalRefresh(Widget: PGtkWidget; user: Pointer); cdecl;
var
Info: PWidgetInfo;
begin
Info := PWidgetInfo(g_object_get_data(PGObject(Widget), 'widgetinfo'));
TTerminal(Info.LCLObject).DoTerminalVisibleChanged;
end;
function TerminalTextScrolled(Widget: PGtkWidget; event: PGdkEvent; user: Pointer): gboolean; cdecl;
var
Info: PWidgetInfo;
begin
result := false;
Info := PWidgetInfo(g_object_get_data(PGObject(Widget), 'widgetinfo'));
if event^._type = 31 then //NOTE: should be 7 for scroll ?
begin
TTerminal(Info.LCLObject).DoTerminalTextScrolled(integer(event^.scroll.direction) and 1);
result := false;
end;
end;
class procedure TGtk2WSTerminal.SetCallbacks(const AGtkWidget: PGtkWidget;
const AWidgetInfo: PWidgetInfo);
begin
TGtk2WSWinControl.SetCallbacks(PGtkObject(AGtkWidget), TComponent(AWidgetInfo^.LCLObject));
end;
class function TGtk2WSTerminal.CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): TLCLIntfHandle;
var
Info: PWidgetInfo;
Style: PGtkRCStyle;
Args: array[0..1] of PChar = (nil, nil);
Allocation: TGTKAllocation;
const
Flgs: array[boolean] of integer = (GTK_CAN_FOCUS, GTK_CAN_FOCUS or GTK_DOUBLE_BUFFERED);
begin
Args[0] := vte_get_user_shell();
if Args[0] = nil then
Args[0] := '/bin/bash';
{ Initialize widget info }
Info := CreateWidgetInfo(gtk_frame_new(nil), AWinControl, AParams);
Info.LCLObject := AWinControl;
Info.Style := AParams.Style;
Info.ExStyle := AParams.ExStyle;
Info.WndProc := {%H-}PtrUInt(AParams.WindowClass.lpfnWndProc);
TTerminal(AWinControl).FInfo := Info;
{ Configure core and client }
gtk_frame_set_shadow_type(PGtkFrame(Info.CoreWidget), GTK_SHADOW_NONE);
Style := gtk_widget_get_modifier_style(Info.CoreWidget);
Style.xthickness := 0;
Style.ythickness := 0;
gtk_widget_modify_style(Info.CoreWidget, Style);
if csDesigning in AWinControl.ComponentState then
Info.ClientWidget := CreateFixedClientWidget(True)
else
begin
Info.ClientWidget := vte_terminal_new();
TTerminal(AWinControl).fTerminalHanlde := VTE_TERMINAL(Info.ClientWidget);
vte_terminal_fork_command_full(VTE_TERMINAL(Info.ClientWidget), VTE_PTY_DEFAULT,
nil, @Args[0], nil, G_SPAWN_SEARCH_PATH, nil, nil, nil, nil);
end;
GTK_WIDGET_SET_FLAGS(Info.CoreWidget, Flgs[AWinControl.DoubleBuffered]);
gtk_container_add(GTK_CONTAINER(Info.CoreWidget), Info.ClientWidget);
g_object_set_data(PGObject(Info.ClientWidget), 'widgetinfo', Info);
gtk_widget_show_all(Info.CoreWidget);
Allocation.X := AParams.X;
Allocation.Y := AParams.Y;
Allocation.Width := AParams.Width;
Allocation.Height := AParams.Height;
gtk_widget_size_allocate(Info.CoreWidget, @Allocation);
g_signal_connect(Info.ClientWidget, 'child-exited', G_CALLBACK(@TerminalExit), nil);
g_signal_connect(Info.ClientWidget, 'contents-changed',G_CALLBACK(@TerminalRefresh), nil);
g_signal_connect(Info.ClientWidget, 'scroll-event', G_CALLBACK(@TerminalTextScrolled), nil);
//g_signal_connect(Info.ClientWidget, 'commit', G_CALLBACK(@TerminalCommit), nil);
g_signal_connect(Info.ClientWidget, 'cursor-moved', G_CALLBACK(@TerminalCommit), nil);
SetCallbacks(Info.CoreWidget, Info);
Result := {%H-}TLCLIntfHandle(Info.CoreWidget);
end;
{$endif}
procedure TTerminal.DoTerminate;
begin
if Assigned(FOnTerminate) then
FOnTerminate(Self);
end;
procedure TTerminal.DoTerminalVisibleChanged;
begin
if Assigned(fOnTerminalVisibleChanged) then
fOnTerminalVisibleChanged(Self);
end;
procedure TTerminal.DoTerminalTextScrolled(delta: integer);
begin
if Assigned(fOnTerminalTextScrolled) then
fOnTerminalTextScrolled(self, delta);
end;
{$ifdef windows}
function ReparentTerminalClbck(wnd: HWND; userp: LPARAM):WINBOOL; stdcall;
var
h: DWORD = 0;
begin
result := true;
with TTerminal(userp) do
begin
GetWindowThreadProcessId(wnd, h);
if (fTermWnd = 0) and (h = fTermProcess.ProcessID) then
begin
fTermWnd := wnd;
windows.SetParent(fTermWnd, Handle);
SetWindowLongPtr(fTermWnd, GWL_STYLE, WS_MAXIMIZE or WS_VISIBLE);
end;
if (fTermWnd <> 0) then
begin
SetWindowPos(fTermWnd, HWND_TOP, 0, 0, width, height, SWP_SHOWWINDOW);
result := false;
end;
end;
end;
{$endif}
procedure TTerminal.Restart;
{$ifdef hasgtk2term}
var
Info: PWidgetInfo;
Args: array[0..1] of PChar = (nil, nil);
{$endif}
begin
{$ifdef hasgtk2term}
if not HandleAllocated then
Exit;
Args[0] := vte_get_user_shell();
if Args[0] = nil then
Args[0] := '/bin/bash';
Info := PWidgetInfo(FInfo);
gtk_widget_destroy(Info.ClientWidget);
Info.ClientWidget := vte_terminal_new;
fTerminalHanlde := VTE_TERMINAL(Info.ClientWidget);
vte_terminal_fork_command_full(fTerminalHanlde, VTE_PTY_DEFAULT,
nil, @Args[0], nil, G_SPAWN_SEARCH_PATH, nil, nil, nil, nil);
gtk_container_add(GTK_CONTAINER(Info.CoreWidget), Info.ClientWidget);
g_object_set_data(PGObject(Info.ClientWidget), 'widgetinfo', Info);
gtk_widget_show_all(Info.CoreWidget);
g_signal_connect(Info.ClientWidget, 'child-exited', G_CALLBACK(@TerminalExit), nil);
g_signal_connect(Info.ClientWidget, 'contents-changed',G_CALLBACK(@TerminalRefresh), nil);
g_signal_connect(Info.ClientWidget, 'scroll-event', G_CALLBACK(@TerminalTextScrolled), nil);
//g_signal_connect(Info.ClientWidget, 'commit', G_CALLBACK(@TerminalCommit), nil);
g_signal_connect(Info.ClientWidget, 'cursor-moved', G_CALLBACK(@TerminalCommit), nil);
{$endif}
{$ifdef Windows}
if assigned(fTermProcess) then
begin
fTermProcess.Terminate(0);
fTermProcess.Free;
end;
fTermProcess := TAsyncProcess.Create(nil);
fTermprocess.Executable:= fTermProgram;
fTermProcess.Execute;
sleep(10);
Reparent();
{$endif}
end;
procedure TTerminal.Reparent;
begin
{$ifdef windows}
if assigned(fTermProcess) then
begin
EnumWindows(@ReparentTerminalClbck, LPARAM(self));
end;
{$endif}
end;
procedure TTerminal.Command(const data: string);
begin
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
vte_terminal_feed_child(fTerminalHanlde, PChar(data + #10), data.Length + 1);
{$endif}
end;
procedure TTerminal.SendControlChar(const cc: TASCIIControlCharacter);
var
c: char;
begin
{$ifdef hasgtk2term}
c := Char(cc);
if assigned(fTerminalHanlde) then
vte_terminal_feed_child(fTerminalHanlde, @c, 1);
{$endif}
end;
procedure TTerminal.copyToClipboard();
begin
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
vte_terminal_copy_clipboard(fTerminalHanlde);
{$endif}
end;
procedure TTerminal.pasteFromClipboard();
begin
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
vte_terminal_paste_clipboard(fTerminalHanlde);
{$endif}
end;
{$ifdef hasgtk2term}
function clbckTrueSel(terminal: PVteTerminal; column: glong; row: glong; data: Pointer): gboolean; cdecl;
begin
result := True;
end;
{$endif}
function TTerminal.getLine(const line: integer): string;
{$ifdef hasgtk2term}
var
c: glong;
a: PGArray = nil;
{$endif}
begin
result := '';
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
c := vte_terminal_get_column_count(fTerminalHanlde);
result := vte_terminal_get_text_range(fTerminalHanlde, line, 0, line, c, @clbckTrueSel, nil, a);
end;
{$endif}
end;
function TTerminal.getWholeText(): string;
{$ifdef hasgtk2term}
var
c: glong;
a: PGArray = nil;
s: string;
i: integer;
{$endif}
begin
result := '';
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
c := vte_terminal_get_column_count(fTerminalHanlde);
for i:= 0 to high(integer) do
begin
s := vte_terminal_get_text_range(fTerminalHanlde, i, 0, i, c, @clbckTrueSel, nil, a);
if (s.Length <> 0) and (s <> #10) then
result += s
else
break;
end;
end;
{$endif}
end;
function TerminalAvailable: Boolean;
begin
{$ifdef hasgtk2term}
Result := Gtk2TermLoad;
{$else}
Result := false;
{$endif}
end;
{$ifdef hasgtk2term}
function RegisterTerminal: Boolean;
begin
Result := TerminalAvailable;
if Result then
RegisterWSComponent(TTerminal, TGtk2WSTerminal);
end;
{$endif}
constructor TTerminal.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
Width := 300;
Height := 200;
fBackgroundColor:= clBlack;
fForegroundColor:= clWhite;
fSelectedColor:= clWhite;
Font.Height:=11;
Font.Name:='Monospace';
fScrollbackLines:=4096;
scrollOnOutut:=true;
fScrollOnKeyStroke:=true;
{$ifdef windows}
fTermProgram := 'cmd.exe';
//Restart;
{$endif}
end;
destructor TTerminal.destroy;
begin
{$ifdef windows}
if assigned(fTermProcess) then
begin
fTermProcess.Terminate(0);
fTermProcess.Free;
end;
{$endif}
inherited;
end;
procedure TTerminal.Paint;
begin
end;
function TTerminal.getCursorPosition: TPoint;
{$ifdef hasgtk2term}
var
col: glong = 0;
row: glong = 0;
{$endif}
begin
{$ifdef hasgtk2term}
result.x:=0;
result.y:=0;
if assigned(fTerminalHanlde) then
begin
vte_terminal_get_cursor_position(fTerminalHanlde, @col, @row);
result.x:= col;
result.y:= row;
end;
{$endif}
end;
procedure TTerminal.setScrollBackLines(value: LongWord);
{$ifdef hasgtk2term}
var
v: TGValue;
begin
fScrollbackLines:=value;
if not assigned(FInFo) then
exit;
v.g_type:= G_TYPE_UINT;
v.data[0].v_uint := fScrollbackLines;
g_object_set_property(PGObject(PWidgetInfo(FInfo).ClientWidget), 'scrollback-lines', @v);
{$else}
begin
{$endif}
end;
procedure TTerminal.setScrollOnOutput(value: boolean);
{$ifdef hasgtk2term}
var
v: TGValue;
begin
fScrollOnOutput:=value;
if not assigned(FInFo) then
exit;
v.g_type:= G_TYPE_UINT;
v.data[0].v_uint := integer(fScrollOnOutput);
g_object_set_property(PGObject(PWidgetInfo(FInfo).ClientWidget), 'scroll-on-output', @v);
{$else}
begin
{$endif}
end;
procedure TTerminal.setScrollOnKeyStroke(value: boolean);
{$ifdef hasgtk2term}
var
v: TGValue;
begin
fScrollOnKeyStroke:=value;
if not assigned(FInFo) then
exit;
v.g_type:= G_TYPE_UINT;
v.data[0].v_uint := integer(fScrollOnKeyStroke);
g_object_set_property(PGObject(PWidgetInfo(FInfo).ClientWidget), 'scroll-on-keystroke', @v);
{$else}
begin
{$endif}
end;
procedure TTerminal.setBackgroundColor(value: TColor);
{$ifdef hasgtk2term}
var
c: TGDKColor;
{$endif}
begin
fBackgroundColor:=value;
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
c := TColortoTGDKColor(fBackgroundColor);
vte_terminal_set_color_background(fTerminalHanlde, @c);
end;
{$endif}
end;
procedure TTerminal.setForegroundColor(value: TColor);
{$ifdef hasgtk2term}
var
c: TGDKColor;
{$endif}
begin
fForegroundColor:=value;
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
c := TColortoTGDKColor(fForegroundColor);
vte_terminal_set_color_foreground(fTerminalHanlde, @c);
vte_terminal_set_color_bold(fTerminalHanlde, @c);
end;
{$endif}
end;
procedure TTerminal.setSelectedColor(value: TColor);
{$ifdef hasgtk2term}
//var
//c: TGDKColor;
{$endif}
begin
fSelectedColor:=value;
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
//c := TColortoTGDKColor(InvertColor(fSelectedColor));
//vte_terminal_set_color_highlight(fTerminalHanlde, @c);
end;
{$endif}
end;
procedure TTerminal.FontChanged(Sender: TObject);
begin
inherited;
{$ifdef hasgtk2term}
{$push}{$Hints off}
if assigned(fTerminalHanlde) and (Handle <> INVALID_HANDLE_VALUE) then
vte_terminal_set_font(fTerminalHanlde, PGtkWidget(Handle).style.font_desc);
{$pop}
{$endif}
end;
function TTerminal.getVScrollInfo: TTerminalScrollInfo;
{$ifdef hasgtk2term}
var
a: PGtkAdjustment;
{$endif}
begin
FillChar(result, sizeOf(result), 0);
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
a := vte_terminal_get_adjustment(fTerminalHanlde);
result.min := trunc(a^.lower);
result.max := max(trunc(a^.upper), result.min + 1);
result.value := trunc(a^.value);
result.pageSize := trunc(a^.page_size);
end;
{$endif}
end;
procedure TTerminal.setVScrollPosition(i: integer);
{$ifdef hasgtk2term}
var
a: PGtkAdjustment = nil;
{$endif}
begin
{$ifdef hasgtk2term}
if assigned(fTerminalHanlde) then
begin
a := vte_terminal_get_adjustment(fTerminalHanlde);
gtk_adjustment_set_value(a, i);
end;
{$endif}
end;
{$ifdef hasgtk2term}
initialization
RegisterTerminal;
{$endif}
end.