首 页 网络编程
网页制作 图形图象 操作系统 冲浪宝典
软件教学 认证考试

网络安全 网络办公 行业资讯 评测对比
您当前位置:站长天空 -> 网络编程-> Visual Basic教程
讲述如何开发一个控件,很有价值(二)_delphi教程
作者:网友供稿 点击:0
推荐
西部数码-全国虚拟主机10强!20余项虚拟主机管理功能,全国领先!第6代双线路虚拟主机,南北访问畅通无阻!可在线rar解压,自动数据恢复设置虚拟目录等.免费赠送访问统计,企业邮局.Cn域名注册10元/年,自助建站480元起,免费试用7天,满意再付款!P4主机租用799元/月.月付免压金
站内搜索
文章页数:[1] 

would be divided up into:

procedure tsKeyWord
  tsSpace
TForm1 tsIdentifier
. tsSymbol
FormCreate tsIdentifier
( tsSymbol
Sender tsIdentifier
: tsSymbol
TObject tsIdentifier
); tsSymbol
  tsSpace
{Create Form} tsComment
<CR><LF> tsCRLF

How is it Done?

The RichEdit control normally loads preformatted text from .RTF files by way of by of the RichEdit.Lines.LoadFromFile() function. YourPasEdit uses the RichEdit.Lines.LoadFromStream() function to load the file from a TPasConversion - a custom TMemoryStream descendant. This stream takes the plaint text Pascal source file, loads it into its internal memory buffe, and then converts it from plain text to a text impregnated with RTF codes. This way when it is loaded into the RichEdit control via RichEdit.Lines.LoadFromStream the Pascal source file appears in the control color-syntax highlighted.

To the main Editor, this process is transparent - the code looks something like this:
 

begin

    NewRichEdit := TRichEdit.Create;

    PasCon.Clear;                    // Prepare the TPasConversion 
    PasCon.LoadFromFile(FName);      // Load the File into the Memory Stream
    PasCon.ConvertReadStream;        // Convert the stream to RTF format

    NewRichEdit.Lines.BeginUpdate;
    NewRichEdit.Lines.LoadFromStream(PasCon);   // Read from the TPasConversion
    NewRichEdit.Lines.EndUpdate

    NewRichEdit.Show;

    Result  := NewRichEdit;

end

EXAMPLE - snippet of code from the NewRichEditCreate(Fname) routine

As I said, it is the TMemoryStream derived TPasConversion which does all the hard work:
 

<SOURCE PASCAL FILE>
|
V
Plain source loaded into memory
  (TPasConversion.LoadFromFile)
|
V
Converted internally by parsing the source file
(ConvertReadStream)
|
V
Result made available 
(SetMemoryPointer)
|
V
RichEdit.LoadFromStream

Most of the work in TPasConversion is done by the ConvertReadStream procedure. Its purpose is to split up each line of source code into tokens (as showed previously) and then depending on its TokenType, load it into the outbuffer preceded by RTF codes to make it a particular Color, Bold, Italics etc. Here what it looks like:
 

// prepare the Outbuf to a certain default size

FOutBuffSize:= size+3;
ReAllocMem(FOutBuff, FOutBuffSize);

// Initialise the parser to its begining state

FTokenState  := tsUnknown;
FComment     := csNo;
FBuffPos     := 0;
FReadBuff    := Memory;

// Write leading RTF Header

WriteToBuffer({ tf1ansideff0deftab720{fonttbl{f0fswiss MS SansSerif;}
{f1fromanfcharset2 Symbol;}{f2fmodern Courier New;}}+#13+#10);
WriteToBuffer({colortbl ed0green0lue0;}+#13+#10);
WriteToBuffer(deflang1033pardplainf2fs20 );

// Create the INSTREAM (FReadBuff) and tokenize it

Result:= Read(FReadBuff^, Size);
FReadBuff[Result] := #0;

if Result > 0 then
begin

     Run:= FReadBuff;
     TokenPtr:= Run;

     while Run^ <> #0 do
     begin

          Case Run^ of

          #13:                        // Deal with CRLFs
          begin
             FComment:= csNo;
             HandleCRLF;
          end;

          #1..#9, #11, #12, #14..#32: // Deal with various whitespaces, control codes
          begin
             while Run^ in [#1..#9, #11, #12, #14..#32] do inc(Run);
             FTokenState:= tsSpace;
             TokenLen:= Run - TokenPtr;
             SetString(TokenStr, TokenPtr, TokenLen);
             SetRTF;
             WriteToBuffer(Prefix + TokenStr + Postfix);
             TokenPtr:= Run;
          end;

          ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
           ~~~~~  much code removed   ~~~~~~~~~~
          ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

     end;

end

EXAMPLE - snippet showing the while loop that breaks up the INSTREAM into recognised tokens

Most of the work is done by the [case Run^ in ... end;] section which "breaks off" a token from the INSTREAM (FReadBuf) based on the logic in the case statement. The case statement is organised in such a way that it can quickly decipher the input stream into the various TokenTypes by examining each character in turn. Having worked out which tokentype it is,  the actual encoding part is relatively easy:
 

FTokenState:= tsSpace;
TokenLen:= Run - TokenPtr;
SetString(TokenStr, TokenPtr, TokenLen);
ScanForRTF;
SetRTF;
WriteToBuffer(Prefix + TokenStr + Postfix);
  EXAMPLE - basic steps in encoding the output stream of the TPasConversion object 

What’s happening here is the program:

  • sets FTokenState to what we believe it is (in this part of the code it is tsSpace which matches any series of Whitespaces)
  • the length of the token is calculated by working out how far the current memory pointer (Run) has moved since we finished with the last token (TokenPtr).
  • the token is then copied from the Input buffer from the starting position of it in the memory buffer (TokenPtr) for the length of the token, into the variable TokenStr.
  • ScanForRtf just checks through the resultant TokenStr to ensure it doesnt have any funny characters that the RichEdit would confuse as RTF commands. If it finds any, it escapes them out.
  • SetRTF looks at the FTokenState to populate two global variables Prefix and Postfix with the appropriate RTF codes to give the token the right Color,Font,Boldness.
  • WriteToBuffer than simply puts the TokenStr with the Prefix and Postfix around it into the output buffer, and the loop then continues on.

Back to the topic: Syntax Highlighting (on-the-fly)

No source code is necessarily 100% applicable to your needs. I was fortunate in that most of the parser applied to my 4GL command syntax (e.g Strings were strings, Numbers were numbers, similar Keywords). As well YourPasEditor had implemented most of the basic accessory tasks such as Printing, Find, Find and Replace, Multi-File editing. It was just a matter of adding in the extras I was after.

PROBLEM #1 - No colours or fonts

One task the Parser didnt fully implement was Colors or Different Fonts, or even fonts sizes. The reason for this (after some trial and error) was that the SetRTF procedure new nothing about how to do this. It only used the information in regards [Bold], [Italics] and [Underline] stored in the Win95 Registry for the Delphi Editors Settings to determine how to highlight each token. As for fonts - well I hadnt realised that the Delphi Editor actually uses only one Font and Fontsize for all the different tokens - so that wasnt Pas2Rtf fault. I was just being greedy.

Luckily the comments in Pas2Rtt.pas told me what the other values in the Registry coded for, especially where the important foreground color was stored. This meant some changes to:

1. procedure SetDelphiRTF(S: String; aTokenState: TTokenState);

        Add after the try;

           Font.Color := StrToInt(Ed_List[0]);

2. procedure TPasConversion.SetPreAndPosFix

        Add after FPreFix[aTokenState] = ;

           FPreFixList[aTokenState] := ColorToRtf(aFont.Color);

The ColorToRtf codes is already present, but hadnt been used for some reasone. If you try it out youll understand why :-). You get absolutely no change except lots of ; in the wrong place.Change the ; to (space) in ColorToRtf(), and you get rid of the ; appearing in the RichEdit control, but no Colors anyway.

My first thought was that the value in Ed_List[0] didnt convert to a proper Font.Color. The easiest way to test this was to hard code Font.Color := clGreen; and see what happens. Again no luck. The format was consistent with the RTF codes I could see in the RTF header. What the $#%#$%# was wrong with it ?

It was about then that I realised I needed a crash course in RTF document structure. For this I rushed off to www.microsoft.com (please forgive me) and found a reference on RTF. After an hour of reading a Microsoft Technical Document I was even more confused. Oh well - this meant it was time to get dirty. Time to get down to real programmer stuff. Time to "cheat".

What did I do? I went into WordPad (which is just a glorified RichEdit version 2.0 on steroids) and saved various files into RTF format. I then opened them in NotePad so I could see the RTF codes and compare what happened in each case: what codes were produced depending on what I did, or didnt do. A similar sort of technique was used back in the 1980s to decipher the first Paradox database format :-) Sorry Borland.


文章整理:站长天空 网址:http://www.z6688.com/
以上信息与文章正文是不可分割的一部分,如果您要转载本文章,请保留以上信息,谢谢!

文章页数:[1] 


放大字体显示 缩小字体显示 打印文章 推荐给朋友
热门文章
·ASP.net中动态加载控件时一些问题的总结-ASP教程,ASP应用
·C#读写注册表-.NET教程,C#语言
·Visual Basic .NET中的异常处理简介(下)-.NET教程,VB.Net语言
·C#:文件的按行读/写及文件目录对话框的使用-.NET教程,C#语言
·ADO.Net:使用DataReader向数据库中插入数据-ASP教程,数据库相关
·列一张网恋赔偿清单(爆笑)
·如何用Photoshop画服装款式图-网页设计,Photoshop
·.NET下使用DataAdapter保存数据时,如何生成command语句及使用事务-.NET教程,数据库应用
·新型dc/dc电源控制芯片dpa426的应用
·ASP.NET 2.0 - Enter Key - Default Submit Button-.NET教程,Asp.Net开发
最新文章
·个人站长的网络赚钱两条新出路_网赚技巧
·adsense帐户最佳化纵深谈-adsense资深专员_网赚技巧
·google adsense容易被k的可能性列表_网赚技巧
·如何让程序被站长接受和产生利润_站长访谈
·马云,即成的中国互联网第4代霸主_站长访谈
·google关键词广告创建的十二招_google推广
·如何使google更快速收录你的新站_google推广
·几个颇有创意的网站推广方法_站长心得
·网络编辑:标题,如何让网民一见钟情(2)_网络编辑
·网站建设基础seo搜索引擎优化_seo网站优化
相关主题
  • 讲述如何开发一个控件,很有价值(七)_delphi教程
  • 讲述如何开发一个控件,很有价值(六)_delphi教程
  • 讲述如何开发一个控件,很有价值(五)_delphi教程
  • 讲述如何开发一个控件,很有价值_delphi教程
  • 西部数码虚拟主机

    友情链接
    CNNIC 西部数码
    万网 自助建站
    虚拟主机 asp空间
    域名注册 域名
    域名申请 主页空间
    论坛空间 网站空间
    国际域名 虚拟空间
    空间租用 DDOS防火墙
    成都主机托管 四川主机托管
    主机租用 服务器租用
    网站目录 自助建站
    虚拟主机 网址大全
    软件下载
    自助链接
    虚拟主机资讯 特价虚拟主机
    版权申明:本站文章均来自网络,如有侵权,请联系我们,我们收到后立即删除,谢谢!
    关于我们:站长天空:专业提供最新的站长资讯、在线教程、虚拟主机权威评测、虚拟主机性能对比、网站制作教程,开发教程,站长工具。包括网页制作教程、冲浪宝典、编程参考、操作系统、软件教学、行业动态等。
    特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有。
    发表评论 打印  刷新     关闭