gstk_editor.erl
来自「OTP是开放电信平台的简称」· ERL 代码 · 共 394 行
ERL
394 行
%% ``The contents of this file are subject to the Erlang Public License,%% Version 1.1, (the "License"); you may not use this file except in%% compliance with the License. You should have received a copy of the%% Erlang Public License along with this software. If not, it can be%% retrieved via the world wide web at http://www.erlang.org/.%% %% Software distributed under the License is distributed on an "AS IS"%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See%% the License for the specific language governing rights and limitations%% under the License.%% %% The Initial Developer of the Original Code is Ericsson Utvecklings AB.%% Portions created by Ericsson are Copyright 1999, Ericsson Utvecklings%% AB. All Rights Reserved.''%% %% $Id$%%%% ------------------------------------------------------------%% Basic Editor Type%% -------------------------------------------------------------module(gstk_editor).%%------------------------------------------------------------------------------%% CANVAS OPTIONS%%%% Attributes:%% activebg Color%% anchor n,w,s,e,nw,se,ne,sw,center%% bc Color%% bg Color%% bw Wth%% data Data%% fg Color%% font Font%% height Int%% highlightbg Color%% highlightbw Wth%% highlightfg Color%% hscroll Bool | top | bottom%% insertbg Color%% insertbw Wth%% insertpos {Row,Col}|'end' (Row: 1..Max, Col: 0..Max)%% justify left|right|center%% padx Int (Pixels)%% pady Int (Pixels)%% relief Relief%% scrollbg Color%% scrollfg Color%% selectbg Color%% selectbw Width%% selectfg Color%% vscroll Bool | left | right%% width Int%% wrap none | char | word%% x Int%% y Int%%%%%% Commands:%% clear%% del {FromIdx, ToIdx} %% enable Bool%% file String%% get {FromIdx, ToIdx} => Text%% insert {Index, Text}Index = [insert,{Row,lineend},end,{Row,Col}]%% setfocus Bool%%%% Events:%% buttonpress [Bool | {Bool, Data}]%% buttonrelease [Bool | {Bool, Data}]%% destroy [Bool | {Bool, Data}]%% enter [Bool | {Bool, Data}]%% focus [Bool | {Bool, Data}]%% keypress [Bool | {Bool, Data}]%% keyrelease [Bool | {Bool, Data}]%% leave [Bool | {Bool, Data}]%% motion [Bool | {Bool, Data}]%%%% Read Options:%% children%% id%% parent%% type%%%.t tag names 2.7 -> red blue (blue 鋜 f鋜gen)%.t tag add blue 2.1 2.10 tagga text%.t tag configure blue -foregr blue skapa tag% .t index end -> MaxRows.cols% .t yview moveto (Row-1)/MaxRows-export([create/3, config/3, read/3, delete/2,event/5,option/5,read_option/5]).-include("gstk.hrl").%%-----------------------------------------------------------------------------%% MANDATORY INTERFACE FUNCTIONS%%-----------------------------------------------------------------------------%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Function : create/3%% Purpose : Create a widget of the type defined in this module.%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%create(DB, Gstkid, Opts) -> MainW = gstk_generic:mk_tkw_child(DB,Gstkid), Editor = lists:append(MainW,".z"), {Vscroll, Hscroll, NewOpts} = gstk_generic:parse_scrolls(Opts), WidgetD = #so{main=MainW, object=Editor, hscroll=Hscroll, vscroll=Vscroll,misc=[{1,white}]}, NGstkid=Gstkid#gstkid{widget=MainW, widget_data=WidgetD}, gstk_db:insert_widget(DB,NGstkid), MandatoryCmd = ["so_create text ", MainW], case gstk:call(MandatoryCmd) of {result, _} -> SimplePreCmd = [MainW, " conf"], PlacePreCmd = [";place ", MainW], case gstk_generic:make_command(NewOpts, NGstkid, MainW, SimplePreCmd, PlacePreCmd, DB,Editor) of {error,Reason} -> {error,Reason}; Cmd -> gstk:exec(Cmd), gstk:exec( [Editor," conf -bo 2 -relief sunken -highlightth 2;", MainW,".sy conf -rel sunken -bo 2;", MainW,".pad.sx conf -rel sunken -bo 2;", Editor, " tag co c1 -for white;"]), ok end end.%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Function : config/3%% Purpose : Configure a widget of the type defined in this module.%% Args : DB - The Database%% Gstkid - The gstkid of the widget%% Opts - A list of options for configuring the widget%%%% Return : [true | {bad_result, Reason}]%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%config(DB, Gstkid, Options) -> SO = Gstkid#gstkid.widget_data, MainW = Gstkid#gstkid.widget, Editor = SO#so.object, NewOpts = case {gs:assq(vscroll,Options),gs:assq(hscroll,Options)} of {false,false} -> Options; _ -> gstk_generic:parse_scrolls(Gstkid, Options) end, SimplePreCmd = [MainW, " conf"], PlacePreCmd = [";place ", MainW], gstk_generic:mk_cmd_and_exec(NewOpts, Gstkid, MainW, SimplePreCmd, PlacePreCmd, DB, Editor).%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Function : read/3%% Purpose : Read one option from a widget%% Args : DB - The Database%% Gstkid - The gstkid of the widget%% Opt - An option to read%%%% Return : [OptionValue | {bad_result, Reason}]%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%read(DB, Gstkid, Opt) -> SO = Gstkid#gstkid.widget_data, gstk_generic:read_option(DB, Gstkid, Opt,SO#so.object).%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Function : delete/2%% Purpose : Delete widget from databas and return tkwidget to destroy%% Args : DB - The Database%% Gstkid - The gstkid of the widget%%%% Return : TkWidget to destroy%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%delete(DB, Gstkid) -> gstk_db:delete_widget(DB, Gstkid), Gstkid#gstkid.widget.event(DB, Gstkid, Etype, Edata, Args) -> gstk_generic:event(DB, Gstkid, Etype, Edata, Args).%%-----------------------------------------------------------------------------%% MANDATORY FUNCTIONS%%-----------------------------------------------------------------------------%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Function : option/5%% Purpose : Take care of options%% Args : Option - An option tuple%% Gstkid - The gstkid of the widget%% MainW - The main tk-widget%% Editor - The Editor tk-widget%% DB - The Database%%%% Return : A tuple {OptionType, OptionCmd}%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%option(Option, Gstkid, _MainW, DB, Editor) -> case Option of {font,Font} when is_tuple(Font) -> gstk_db:insert_opt(DB,Gstkid,Option), {c, [Editor, " conf -font ", gstk_font:choose_ascii(DB,Font)]}; {font_style, {{Start,End},Font}} -> % should be only style {Tag,Ngstkid} = get_style_tag(DB,Editor,Font,Gstkid), gstk_db:update_widget(DB,Ngstkid), {c, Ngstkid, [Editor, " tag ad ", Tag, " ", p_index(Start), " ", p_index(End)]}; {fg, {{Start,End},Color}} -> {Tag,Ngstkid} = get_color_tag(Editor,Color,Gstkid), gstk_db:update_widget(DB,Ngstkid), {c, Ngstkid, [Editor, " tag ad ", Tag, " ", p_index(Start), " ", p_index(End)]}; {padx, Pad} -> {c, [Editor," conf -padx ",gstk:to_ascii(Pad)]}; {pady, Pad} -> {c, [Editor," conf -pady ",gstk:to_ascii(Pad)]}; {selection, {From, To}} -> {c, [Editor," tag ad sel ",p_index(From)," ", p_index(To)]}; {vscrollpos, Row} -> {MaxRow,_Col} = ret_ed_index([Editor," ind end"]), {c, [Editor, " yv mo ",gstk:to_ascii(Row/MaxRow)]}; {wrap, How} -> {c, [Editor, " conf -wrap ", gstk:to_ascii(How)]}; {fg, Color} -> {c, [Editor, " conf -fg ", gstk:to_color(Color)]}; {insertbw, Wth} -> {c, [Editor, " conf -insertbo ", gstk:to_ascii(Wth)]}; {insertbg, Color} -> {c, [Editor, " conf -insertba ", gstk:to_color(Color)]}; {insertpos, Index} -> {c, [Editor, " m s insert ", p_index(Index)]}; {insert, {Index, Text}} -> {c, [Editor, " ins ", p_index(Index), " ", gstk:to_ascii(Text)]}; {del, {From, To}} -> {c, [Editor, " del ", p_index(From), " ", p_index(To)]}; {overwrite, {Index, Text}} -> AI = p_index(Index), Len = gstk:to_ascii(lists:flat_length(Text)), {c, [Editor, " del ",AI," \"",AI,"+",Len,"c\";", Editor, " ins ",AI," ", gstk:to_ascii(Text)]}; clear -> {c, [Editor, " delete 1.0 end"]}; {load, File} -> {ok, F2,_} = regexp:gsub(File, [92,92], "/"), case gstk:call(["ed_load ", Editor, " ", gstk:to_ascii(F2)]) of {result, _} -> none; {bad_result,Re} -> {error,{no_such_file,editor,load,F2,Re}} end; {save, File} -> {ok, F2,_} = regexp:gsub(File, [92,92], "/"), case gstk:call(["ed_save ",Editor," ",gstk:to_ascii(F2)]) of {result, _} -> none; {bad_result,Re} -> {error,{no_such_file,editor,save,F2,Re}} end; {enable, true} -> {c, [Editor, " conf -sta normal"]}; {enable, false} -> {c, [Editor, " conf -sta disabled"]}; {setfocus, true} -> {c, ["focus ", Editor]}; {setfocus, false} -> {c, ["focus ."]}; _ -> invalid_option end.%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Function : read_option/5%% Purpose : Take care of a read option%% Return : The value of the option or invalid_option%% [OptionValue | {bad_result, Reason}]%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%read_option(Option,GstkId,_MainW,DB,Editor) -> case Option of font -> gstk_db:opt(DB,GstkId,font,undefined); padx -> tcl2erl:ret_atom([Editor," cg -padx"]); pady -> tcl2erl:ret_atom([Editor," cg -pady"]); enable -> tcl2erl:ret_enable([Editor," cg -st"]); fg -> tcl2erl:ret_color([Editor," cg -fg"]); {fg, Pos} -> L=tcl2erl:ret_list([Editor," tag nam ", p_index(Pos)]), SO = GstkId#gstkid.widget_data, case last_tag_val(undefined, $c, L, SO#so.misc) of undefined -> tcl2erl:ret_color([Editor," cg -fg"]); Color -> Color end; {font_style, Pos} -> L=tcl2erl:ret_list([Editor," tag nam ", p_index(Pos)]), SO = GstkId#gstkid.widget_data, case last_tag_val(undefined, $f, L, SO#so.misc) of undefined -> 'my style? nyi'; Style -> Style end; selection -> ret_ed_indexes([Editor," tag ne sel 1.0"]); char_height -> tcl2erl:ret_int([Editor, " cg -he"]); char_width -> tcl2erl:ret_int([Editor, " cg -wi"]); insertbg -> tcl2erl:ret_color([Editor," cg -insertba"]); insertbw -> tcl2erl:ret_int([Editor," cg -insertbo"]); insertpos -> ret_ed_index([Editor, " ind insert"]); setfocus -> tcl2erl:ret_focus(Editor, "focus"); wrap -> tcl2erl:ret_atom([Editor," cg -wrap"]); size -> {MaxRow,_Col} = ret_ed_index([Editor," ind end"]), MaxRow-1; vscrollpos -> {MaxRow,_Col} = ret_ed_index([Editor," ind end"]), [Top,_Bot] = tcl2erl:ret_list([Editor," yvi"]), round(Top*(MaxRow-1))+1; {get, {From, To}} -> tcl2erl:ret_str([Editor, " get ", p_index(From), " ", p_index(To)]); _ -> {bad_result, {GstkId#gstkid.objtype, invalid_option, Option}} end.%%------------------------------------------------------------------------------%% PRIMITIVES%%------------------------------------------------------------------------------p_index({Line, lineend}) -> [$",gstk:to_ascii(Line), ".1 lineend",$"];p_index({Line, Char}) -> [gstk:to_ascii(Line), $., gstk:to_ascii(Char)];p_index(insert) -> "insert";p_index('end') -> "end";p_index(Idx) -> gs:error("bad index in editor: ~w~n",[Idx]),0.ret_ed_index(Cmd) -> case gstk:call(Cmd) of {result, Val} -> case io_lib:fread("~d.~d", Val) of {ok, [Row,Col], []} -> {Row, Col}; Other -> {bad_result, Other} end; Bad_result -> Bad_result end.ret_ed_indexes(Cmd) -> case gstk:call(Cmd) of {result, ""} -> undefined; {result, Val} -> case io_lib:fread("~d.~d ~d.~d", Val) of {ok, [Row1,Col1,Row2,Col2], []} -> {{Row1, Col1}, {Row2,Col2}}; Other -> {bad_result, Other} end; Bad_result -> Bad_result end.%%----------------------------------------------------------------------%% Returns: {Tag text(), NewGstkId}%%----------------------------------------------------------------------%% The misc field of the so record is a list of {ColorNo, Color|Font|...}get_color_tag(Editor,Color,Gstkid) -> SO = Gstkid#gstkid.widget_data, Tags = SO#so.misc, case lists:keysearch(Color, 2, Tags) of% {value, {No, _}} -> {["c",gstk:to_ascii(No)], Gstkid};% false -> % don't reuse tags, priority order spoils that _Any -> {No,_} = lists:max(Tags), N=No+1, SO2 = SO#so{misc=[{N,Color}|Tags]}, TagStr=["c",gstk:to_ascii(N)], gstk:exec([Editor," tag co ",TagStr," -for ", gstk:to_color(Color)]), {TagStr,Gstkid#gstkid{widget_data=SO2}} end.get_style_tag(DB,Editor,Style,Gstkid) -> SO = Gstkid#gstkid.widget_data, Tags = SO#so.misc, case lists:keysearch(Style, 2, Tags) of% {value, {No, _}} -> {["f",gstk:to_ascii(No)], Gstkid};% false -> % don't reuse tags, priority order spoils that _Any -> {No,_} = lists:max(Tags), N=No+1, SO2 = SO#so{misc=[{N,Style}|Tags]}, TagStr=["f",gstk:to_ascii(N)], gstk:exec([Editor," tag co ",TagStr," -font ", gstk_font:choose_ascii(DB,Style)]), % should be style only {TagStr,Gstkid#gstkid{widget_data=SO2}} end.%%----------------------------------------------------------------------%% Purpose: Given a list of tags for a char, return its visible color%% (that is that last color tag in the list).%%----------------------------------------------------------------------last_tag_val(TagVal, _Chr, [], _TagDict) -> TagVal;last_tag_val(TagVal, Chr, [Tag|Ts],TagDict) -> case atom_to_list(Tag) of [Chr|ANo] -> No = list_to_integer(ANo), last_tag_val(gs:val(No, TagDict),Chr,Ts,TagDict); _NoAcolor -> last_tag_val(TagVal,Chr, Ts,TagDict) end. %%% ----- Done -----
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?