lazarus/ide/charactermapdlg.pas
2020-12-29 22:56:40 +00:00

417 lines
12 KiB
ObjectPascal

{
/***************************************************************************
charactermapdlg.pas
-------------------
***************************************************************************/
***************************************************************************
* *
* This source is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This code is distributed in the hope that it will be useful, but *
* WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
* General Public License for more details. *
* *
* A copy of the GNU General Public License is available on the World *
* Wide Web at <http://www.gnu.org/copyleft/gpl.html>. You can also *
* obtain it by writing to the Free Software Foundation, *
* Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1335, USA. *
* *
***************************************************************************
Author: Mattias Gaertner
Abstract:
Dialog for character map.
}
unit CharacterMapDlg;
{$mode objfpc}{$H+}
interface
uses
{$ifdef WINDOWS}Windows,{$endif}
Classes, SysUtils, Math,
// LCL
Controls, Graphics, Dialogs, Buttons, StdCtrls, Forms,
LCLType, LCLUnicodeData, Grids, ButtonPanel, ComCtrls,
// LazUtils
GraphType, LazUTF8, LConvEncoding,
// IdeIntf
IDEHelpIntf, IDEImagesIntf,
// IDE
LazarusIDEStrConsts, EditorOptions, EnvironmentOpts;
type
TOnInsertCharacterEvent = procedure (const C: TUTF8Char) of object;
{ TCharacterMapDialog }
TCharacterMapDialog = class(TForm)
ButtonPanel: TButtonPanel;
cbCodePage: TComboBox;
AnsiCharInfoLabel: TLabel;
cbUniRange: TComboBox;
SortUniRangeListButton: TSpeedButton;
CodePageLabel: TLabel;
RangeLabel: TLabel;
UnicodeCharInfoLabel: TLabel;
PageControl1: TPageControl;
AnsiGrid: TStringGrid;
UnicodeGrid: TStringGrid;
pgAnsi: TTabSheet;
pgUnicode: TTabSheet;
procedure GridPrepareCanvas(sender: TObject; {%H-}aCol, {%H-}aRow: Integer;
{%H-}aState: TGridDrawState);
procedure cbCodePageSelect(Sender: TObject);
procedure cbUniRangeSelect(Sender: TObject);
procedure FormKeyDown(Sender: TObject; var Key: Word; {%H-}Shift: TShiftState);
procedure HelpButtonClick(Sender: TObject);
procedure FormCreate(Sender: TObject);
procedure FormShow(Sender: TObject);
procedure SortUniRangeListButtonClick(Sender: TObject);
procedure AnsiGridSelectCell(Sender: TObject; aCol, aRow: Integer;
var {%H-}CanSelect: Boolean);
procedure UnicodeGridSelectCell(Sender: TObject; aCol, aRow: Integer;
var {%H-}CanSelect: Boolean);
procedure StringGridKeyPress(Sender: TObject; var Key: char);
procedure StringGridMouseDown(Sender: TObject; Button: TMouseButton;
{%H-}Shift: TShiftState; X, Y: Integer);
procedure AnsiGridMouseMove(Sender: TObject; {%H-}Shift: TShiftState; X,
Y: Integer);
procedure UnicodeGridMouseMove(Sender: TObject; {%H-}Shift: TShiftState; X,
Y: Integer);
private
FOnInsertCharacter: TOnInsertCharacterEvent;
FUnicodeBlockIndex: Integer;
procedure DoStatusAnsiGrid(ACol, ARow: integer);
procedure DoStatusUnicodeGrid(ACol, ARow: integer);
procedure FillAnsiGrid;
procedure FillUnicodeGrid;
procedure FillUniRangeList(ASorted: Boolean);
function UnicodeBlockIndexByName(AName: String): Integer;
function UnicodeBlockSelected: Boolean;
procedure SelectSystemCP;
public
property OnInsertCharacter: TOnInsertCharacterEvent read FOnInsertCharacter
write FOnInsertCharacter;
end;
procedure ShowCharacterMap(AOnInsertChar: TOnInsertCharacterEvent);
var
CharacterMapDialog: TCharacterMapDialog;
implementation
{$R *.lfm}
const
NOT_SELECTED=Low(UnicodeBlocks)-1;
procedure ShowCharacterMap(AOnInsertChar: TOnInsertCharacterEvent);
begin
if CharacterMapDialog = nil then
Application.CreateForm(TCharacterMapDialog, CharacterMapDialog);
CharacterMapDialog.OnInsertCharacter := AOnInsertChar;
CharacterMapDialog.Show;
end;
{ TCharacterMapDialog }
procedure TCharacterMapDialog.FormCreate(Sender: TObject);
begin
Caption := lisCharacterMap;
RangeLabel.Caption := lisRange;
SortUniRangeListButton.Flat:=True;
SortUniRangeListButton.Hint:=lisSortUnicodeRangeListAlphabetically;
IDEImages.AssignImage(SortUniRangeListButton, 'pkg_sortalphabetically');
ButtonPanel.HelpButton.Caption:=lisMenuHelp;
ButtonPanel.CloseButton.Caption:=lisBtnClose;
//EnvironmentOptions.IDEWindowLayoutList.Apply(Self, Name);
PageControl1.ActivePageIndex := 0;
AnsiCharInfoLabel.Caption := '-';
UnicodeCharInfoLabel.Caption := '-';
SelectSystemCP;
FillAnsiGrid;
end;
procedure TCharacterMapDialog.SelectSystemCP;
{$ifdef Windows}
var
i: Integer;
cp: Word;
cpStr: String;
{$endif}
begin
{$ifdef Windows}
// Find system code page on Windows...
// see: msdn.microsoft.com/library/windows/desktop/dd317756%28v=vs.85%29.aspx
cp := Windows.GetACP;
case cp of // add spaces to be sure of unique names found in the combobox
437..1258: cpStr := 'cp' + IntToStr(cp) + ' ';
10000 : cpStr := 'macintosh ';
20866 : cpStr := 'koi8 ';
28591 : cpStr := 'iso88591 ';
28592 : cpStr := 'iso88592 ';
28605 : cpStr := 'iso885915 ';
else cpStr := '';
end;
for i := 0 to cbCodePage.Items.Count-1 do
if pos(cpStr, cbCodePage.Items[i]) = 1 then
begin
cbCodePage.ItemIndex := i;
exit;
end;
{$endif}
// ... if not found, or non-Windows, just pick the first item.
cbCodePage.ItemIndex := 0;
end;
procedure TCharacterMapDialog.HelpButtonClick(Sender: TObject);
begin
LazarusHelp.ShowHelpForIDEControl(Self);
end;
function RoundUp(Value, Divi:integer):integer;
begin
if Value mod Divi = 0 then
Result:=Value div Divi
else
Result:=(Value div Divi)+1;
end;
procedure TCharacterMapDialog.cbCodePageSelect(Sender: TObject);
begin
FillAnsiGrid;
end;
procedure TCharacterMapDialog.cbUniRangeSelect(Sender: TObject);
begin
FUnicodeBlockIndex:=UnicodeBlockIndexByName(cbUniRange.Text);
FillUnicodeGrid;
end;
procedure TCharacterMapDialog.FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
begin
if Key=VK_ESCAPE then
begin
Close;
Key:= 0;
end;
end;
procedure TCharacterMapDialog.FormShow(Sender: TObject);
begin
AnsiGrid.Font.Name := EditorOpts.EditorFont;
UnicodeGrid.Font.Name := EditorOpts.EditorFont;
AnsiGrid.Font.Size := 10;
UnicodeGrid.Font.Size := 10;
AnsiGrid.AutoSizeColumn(0);
AnsiGrid.AutoFillColumns := true;
FUnicodeBlockIndex:=NOT_SELECTED;
FillUniRangeList(SortUniRangeListButton.Down);
FillUnicodeGrid;
cbCodePage.DropDownCount := Math.max(EnvironmentOptions.DropDownCount, 25);
cbUniRange.DropDownCount := Math.max(EnvironmentOptions.DropDownCount, 25);
end;
procedure TCharacterMapDialog.SortUniRangeListButtonClick(Sender: TObject);
begin
FillUniRangeList(SortUniRangeListButton.Down);
end;
procedure TCharacterMapDialog.AnsiGridSelectCell(Sender: TObject; aCol,
aRow: Integer; var CanSelect: Boolean);
begin
DoStatusAnsiGrid(aCol, aRow);
end;
procedure TCharacterMapDialog.UnicodeGridSelectCell(Sender: TObject; aCol,
aRow: Integer; var CanSelect: Boolean);
begin
DoStatusUnicodeGrid(aCol, aRow);
end;
procedure TCharacterMapDialog.StringGridKeyPress(Sender: TObject; var Key: char);
var
sg: TStringGrid;
s: string;
begin
if Key = #13 then
begin
sg := Sender as TStringGrid;
s := sg.Cells[sg.Col, sg.Row];
if (s <> '') and (Assigned(OnInsertCharacter)) then
OnInsertCharacter(s);
end;
end;
procedure TCharacterMapDialog.StringGridMouseDown(Sender: TObject;
Button: TMouseButton; Shift: TShiftState; X, Y: Integer);
var
Row, Col: Integer;
sg: TStringGrid;
begin
sg := Sender as TStringGrid;
if (Button = mbLeft) and (sg.MouseToGridZone(X, Y) = gzNormal) then
begin
Col:=0; Row:=0;
sg.MouseToCell(X, Y, Col, Row);
if (sg.Cells[Col, Row] <> '') and (Assigned(OnInsertCharacter)) then
OnInsertCharacter(sg.Cells[Col, Row]);
end;
end;
procedure TCharacterMapDialog.DoStatusAnsiGrid(ACol, ARow: integer);
var
N: integer;
begin
N := ACol-1 + (ARow-1)*16 + 32;
AnsiCharInfoLabel.Caption := Format('Decimal: %s, Hex: $%s', [IntToStr(N), IntToHex(N, 2)]);
end;
procedure TCharacterMapDialog.AnsiGridMouseMove(Sender: TObject;
Shift: TShiftState; X, Y: Integer);
var
Row, Col: Integer;
begin
if AnsiGrid.MouseToGridZone(X, Y) = gzNormal then
begin
Col:=0; Row:=0;
AnsiGrid.MouseToCell(X, Y, Col, Row);
DoStatusAnsiGrid(Col, Row);
end
else
AnsiCharInfoLabel.Caption := '-';
end;
procedure TCharacterMapDialog.GridPrepareCanvas(sender: TObject; aCol,
aRow: Integer; aState: TGridDrawState);
var
ts: TTextStyle;
begin
with (Sender as TStringGrid) do begin
ts := Canvas.TextStyle;
ts.Alignment := taCenter;
Canvas.TextStyle := ts;
end;
end;
procedure TCharacterMapDialog.DoStatusUnicodeGrid(ACol, ARow: integer);
var
S: Cardinal;
tmp, tmp2: String;
i: Integer;
begin
if not UnicodeBlockSelected then Exit;
S:=UnicodeBlocks[FUnicodeBlockIndex].S+(ACol)+(ARow*16);
tmp:=UnicodeToUTF8(S);
tmp2:='';
for i:=1 to Length(tmp) do
tmp2:=tmp2+'$'+IntToHex(Ord(tmp[i]),2);
UnicodeCharInfoLabel.Caption:='U+'+inttohex(S,4)+', UTF-8: '+tmp2;
end;
procedure TCharacterMapDialog.UnicodeGridMouseMove(Sender: TObject;
Shift: TShiftState; X, Y: Integer);
var
Row, Col: Integer;
begin
if UnicodeGrid.MouseToGridZone(X, Y) = gzNormal then
begin
Col:=0; Row:=0;
UnicodeGrid.MouseToCell(X, Y, Col, Row);
DoStatusUnicodeGrid(Col, Row);
end
else
AnsiCharInfoLabel.Caption := '-';
end;
procedure TCharacterMapDialog.FillAnsiGrid;
var
R, C, p: Integer;
cp: String;
begin
cp := cbCodePage.Items[cbCodePage.ItemIndex];
p := pos(' ', cp);
if p > 0 then SetLength(cp, p-1);
for R := 0 to Pred(AnsiGrid.RowCount) do
begin
if R <> 0 then AnsiGrid.Cells[0, R] := Format('%.3d +', [Succ(R) * 16]);
for C := 1 to Pred(AnsiGrid.ColCount) do
begin
if R = 0 then AnsiGrid.Cells[C, R] := Format('%.2d', [Pred(C)])
else
AnsiGrid.Cells[C, R] := ConvertEncoding(Chr(Succ(R) * 16 + Pred(C)), cp, 'utf8');
end;
end;
end;
procedure TCharacterMapDialog.FillUnicodeGrid;
var
cnt, x, y: integer;
S, E: integer;
begin
UnicodeGrid.Clear;
if not UnicodeBlockSelected then
Exit;
S:=UnicodeBlocks[FUnicodeBlockIndex].S;
E:=UnicodeBlocks[FUnicodeBlockIndex].E;
UnicodeGrid.ColCount:=16;
UnicodeGrid.RowCount:=RoundUp(E-S,16);
cnt:=0;
for y:=0 to UnicodeGrid.RowCount-1 do
for x:=0 to UnicodeGrid.ColCount-1 do
begin
if S+Cnt<=E then
UnicodeGrid.Cells[x,y]:=UnicodeToUTF8(S+Cnt);
inc(cnt);
end;
UnicodeGrid.AutoSizeColumns;
end;
procedure TCharacterMapDialog.FillUniRangeList(ASorted: Boolean);
var
BlockIdx: Integer;
begin
cbUniRange.Items.Clear;
cbUniRange.Sorted:=ASorted;
for BlockIdx:=Low(UnicodeBlocks) to High(UnicodeBlocks) do
cbUniRange.Items.Append(UnicodeBlocks[BlockIdx].PG);
if not UnicodeBlockSelected then
FUnicodeBlockIndex:=Low(UnicodeBlocks);
cbUniRange.Text:=UnicodeBlocks[FUnicodeBlockIndex].PG;
end;
function TCharacterMapDialog.UnicodeBlockIndexByName(AName: String): Integer;
var
BlockIdx: Integer;
begin
for BlockIdx:=Low(UnicodeBlocks) to High(UnicodeBlocks) do
if UnicodeBlocks[BlockIdx].PG=AName then
Exit(BlockIdx);
Result:=NOT_SELECTED;
end;
function TCharacterMapDialog.UnicodeBlockSelected: Boolean;
begin
Result:=(FUnicodeBlockIndex>=Low(UnicodeBlocks)) and (FUnicodeBlockIndex<=High(UnicodeBlocks));
end;
end.