Текст с эффектами
WEB-сайт: http://daddy.mirgames.ru
Введение.
Вам когда-нибудь хотелось организовать вывод текста, да не простого, а с наворотами. Так вот эта статья как раз про это. Читайте!
Благодарность...
Сайту Daddy.h1.ru (MirGames.ru) за то, что там можно найти информацию по DelphiX и скачать файлы со шрифтами :) Большое спасибо!
… и информация.
Прошу прошения у тех, кто писал на Relort@xaker.ru в последнее время, за то, что не ответил. В течение последнего месяца я не мог «достучатся» до своего ящика. И вот буквально вчера узнал, что тот на время накрылся :( Так что повторите свои отклики, просьбы, пожелания на новый электронный адрес.
Также убедительная просьба присылать интересующие вас темы, и если я что-либо смогу по ним сказать, то обязательно это сделаю, оформив ответы в статьи. ОК?
Часть 1. Базовый класс.
Лирическое отступление: давайте договоримся, что все новые классы и тому подобное мы будем размешать в отдельных файлах. ОК? По крайне мере я собираюсь делать именно так... Итак, создаем новый проект, кидаем на форму TDXDraw, TDXTimer. Теперь создаем новый Unit и обзываем его, например, TextWithEffect_Unit.pas. Приготовления закончены. Приступаем.
Рассуждаем: текст состоит из слов, слова из букв. Вот почему первым созданием в Unit’е будет:
TLetterEffect = record
Visible: Boolean; //видима ли буква
CurTime, Delay: Integer; //...
Scale: Integer; //это и будет нашим эффектом (уменьшение)
end;
TLetterEffectArray = array of TLetterEffect;
PLetterEffectArray = ^TLetterEffectArray;
То есть описание буквы текста и определение массива букв. Сразу за ним пишем:
TTextWithEffect = class
private
FontSurface: TDirectDrawSurface;
TextSurface: TDirectDrawSurface;
FText: String;
FFontSize: Integer;
FState: Integer;
FFileName: String;
LetterEffect: TLetterEffectArray;
procedure SetText(const Value: String);
procedure Error(ErrorType: Integer);
procedure SetFontSize(const Value: Integer);
procedure SetFileName(const Value: String);
public
property Text: String read FText write SetText;
property FontSize: Integer read FFontSize write SetFontSize;
property FileName: String read FFileName write SetFileName;
procedure Draw(Surface: PDirectDrawSurface);
constructor Create(DD: PDirectDraw);
destructor Destroy;
end;
Надеюсь здесь все понятно? Нет!? Ну тогда ладно:
FontSurface – поверхность (TDirectDrawSurface), содержащая алфавит.
TextSurface - поверхность (TDirectDrawSurface), содержащая текст к выводу.
LetterEffect – буквы текста.
Text – текст к отрисовке.
FontSize – размер каждой буквы текста. Здесь надо оговорится: каждая буква текста берется из файла с именем FileName (третье свойство TTextWithEffect), т. е. FileName – имя файла с алфавитом (см. архив, прилагаемый к статье, файл Font.bmp); и для заполнение TextSurface’а необходим размер каждой буквы.
Ну и так далее...
А сейчас самое интересное: создание эффекта и просмотр результата.
Часть 2. Эффект и отрисовка.
Для данного примера я опишу лишь один эффект – уменьшение букв с течением времени. Этот эффект довольно просто реализовать с учетом вышеописанного класса TLetterEffect. В том классе за текущее состояние эффекта буквы отвечала переменная Scale. Все что нам нужно – уменьшать ее во времени (но до определенного момента).
А вот и сама процедура, принимающая в качестве аргумента указатель на массив букв. Вот...
//Эффект
procedure ScaleDown(L: PLetterEffectArray);
var
i: Integer;
begin
for i:=0 to High(L^) do
begin
if L^[i].Scale < 0 then
begin
inc(L^[i].CurTime);
if L^[i].CurTime <= L^[i].Delay then
dec(L^[i].Scale, 1);
exit;
end;
end;
end;
// ну и отрисовка
procedure TTextWithEffect.Draw(Surface: PDirectDrawSurface);
var
i: Integer;
rc: TRect;
rc1: Trect;
s: Integer;
begin
if FontSurface = nil then exit;
if FText = '' then exit;
// рисуем текст без эффекта
Surface^.Draw(32, 32, TextSurface.ClientRect, TextSurface, True);
// вносим изменения в эффект
ScaleDown(@LetterEffect);
// и выводим текст с эффектом побуквенно
for i:=0 to High(LetterWithEffect) do
begin
s := LetterEffect[i].Scale;
if s = 32 then exit;
rc1 := Rect(i*16,0,(i+1)*16,16);
if s < 0 then
begin
rc := Rect(i*16+32-s, 64-s, ((i+1)*16)+32+s, 80+s);
Surface^.StretchDraw(rc, rc1, TextSurface, True);
end
else Surface^.Draw(i*16+32, 64, rc1, TextSurface, True)
end;
end;
Заключение.
Посмотрите файл TWE_Ex.exe.
Вот чего можно добиться расширив данный пример :) !!!
Ну вот вроде бы как и все. Мы добрались до завершения статьи. Если что непонятно, необходимо пояснение или пример пишите. Отвечу всем.
DelphiWorld 6.0