• Часто задаваемые вопросы по Borland Delphi
  • Общие вопросы по Delphi и данному FAQ (часть 1)
  • Введение
  • Вопросы общего характера
  • Совместимость
  • Базы данных - Interbase и локальные данные
  • Базы данных — прочие SQL сервера
  • Общие вопросы по Delphi и данному FAQ (часть 2)
  • Базы данных — компоненты и VCL.
  • Компоненты и VCL
  • Общие вопросы по Delphi и данному FAQ (часть 3)
  • Object Pascal и Windows API
  • Разное
  • Полезные хитрости
  • Вопросы по Delphi 1.0
  • Вопросы общего характера
  • Delphi и Visual Basic
  • Базы данных
  • Object Pascal и Windows API
  • Компоненты и VCL
  • Разное
  • Вопросы по Delphi 2.0
  • Что нового в Delphi 2.0 по сравнения с Delphi 1.0?
  • Вопросы общего характера
  • Базы данных
  • Object Pascal и Windows API
  • Разное
  • Вопросы по Delphi 3.0
  • Вопросы общего характера
  • Object Pascal и Windows API
  • Delphi, С++ Builder и базы данных: вопросы и ответы
  • Delphi VCL FAQ
  • FAQ

    Часто задаваемые вопросы по Borland Delphi

    Примечание: ответ на интересующий вас вопрос вполне может оказаться в разделе, который дает описание версии, отличной от вашей. Материал не дублируется, где возможно приводятся перекрестные ссылки. Рекомендуется читать все разделы. 

    Условные обозначения, используемые в данном FAQ:

    Вопрос

    Ответ

    Материал относительно Delphi 1.0

    Пример (исходный текст программы) для Delphi 1.0

    Материал относительно Delphi 2.0

    Пример (исходный текст программы) для Delphi 2.0

    Материал относительно Delphi 3.0

    Пример (исходный текст программы) для Delphi 3.0

    Общие вопросы по Delphi и данному FAQ (часть 1)

    Введение

    1. Какова цель этого FAQ?

    Этот FAQ предназначен для ответов на некоторые вопросы относительно новой cреды разработки Borland International, называемой Delphi. Этот документ, в действительности, не FAQ в полном смысле этого термина, потому что некоторые из вопросов, на которые здесь есть ответы, не очень часто задают на самом деле.

    Вначале мы выпускали FAQ, куда помещали все доступные сведения о Delphi в текстовом виде. Далее была промежуточная версия в формате WinHelp. Теперь, после ряда трансформаций FAQ обличен в форму HTML.


    2. Как я могу получить последнюю версию FAQ?

    Проще всего зайти на наш WWW-сервер (www.demo.ru) и загрузить оттуда свежие версии всех имеющихся материалов, которые, кстати, помимо Delphi, охватывают весь спектр продуктов, выпускаемых фирмой Borland International. Также вы можете получить текстовую версию данных материалов, загрузив файл delfaqs.zip.

    Вопросы общего характера

    1. Что такое Delphi?

    Delphi — это достаточно новый продукт Borland International для быстрого создания приложений (RAD). Высокопроизводительный инструмент визуального построения приложений, работающих с базами данных в архитектуре клиент-сервер, Internet/Intranet, а также для локальных машин и файл-серверной архитектуры. Этот инструментарий включает в себя настоящий компилятор кода и предоставляет средства визуального программирования, несколько похожие на те, что можно обнаружить в Microsoft Visual Basic или в других инструментах визуального проектирования. Лежащий в основе Delphi язык — Object Pascal, который является расширением объектно-ориентированного языка Pascal (Turbo/Borland Pascal, начиная с версии 5.5). В Delphi также входят локальный SQL-сервер InterBase 4.0, генераторы отчетов, библиотеки визуальных компонентов, и прочее хозяйство, необходимое для того, чтобы чувствовать себя совершенно уверенным при профессиональной разработке информационных систем или просто программ для Windows-среды. Поскольку в архитектуре клиент-сервер де-факто сложилось такое положение, что клиентские станции работают, как правило, в Windows-среде, а SQL-сервер — в операционной системе UNIX, Delphi Client-Server может служить удобным инструментом для скоростной разработки приложений.


    2. Для кого предназначен Delphi?

    Прежде всего профессиональным разработчикам, желающим очень быстро разрабатывать приложения в архитектуре клиент-сервер. Delphi производит небольшие по размерам (до 15-30 Кбайт в Delphi 3.x!) высокоэффективные исполняемые модули (.exe и .dll), поэтому в Delphi должны быть прежде всего заинтересованы те, кто разрабатывает продукты на продажу. С другой стороны небольшие по размерам и быстро исполняемые модули означают, что требования к клиентским рабочим местам существенно снижаются — это имеет немаловажное значение и для конечных пользователей. Помимо стандартных клиентских приложений (Delphi 1.0 и 2.0), Delphi 3.x также имеет средства для созданий приложений в многозвенной архитектуре.


    3. Преимущества Delphi по сравнению с аналогичными программными продуктами.

    • Быстрота разработки приложения.

    • Высокая производительность разработанного приложения.

    • Hизкие требования разработанного приложения к ресурсам компьютера.

    • Hаращиваемость за счет встраивания новых компонент и инструментов в среду Delphi.

    • Возможность разработки новых компонент и инструментов собственными средствами Delphi (существующие компоненты и инструменты доступны в исходниках)

    • Удачная проработка иерархии объектов

    • Де-факто уже доступно огромное количество визуальных компонентов третьих фирм, часть из которых freeware, часть shareware, часть — коммерческие.


    4. Какие есть версии Delphi?

    В феврале 1995 года была выпущена первая версия Delphi, которая генерировала код, исполняемый под операционной системой Windows 3.1x.

    В начале февраля 1996 года объявлено о выходе второй версии продукта, которая генерирует уже 32-разрядный код для Windows 95 и Windows NT и использует все преимущества 32-разрядных приложений — более высокая скорость обработки данных, большее количество возможностей для приложения и др. Вторая версия Delphi предлагается уже в трех вариантах: Delphi Desktop, Delphi Developer и Delphi Client/Server Suite. Версии Desktop и Developer включают в себя Delphi Desktop 1.0, а Client/Server Suite — Delphi Client/Server 1.0.

    В ближайшем будущем ожидается версия Delphi 3.0, которая также будет создавать 32-разрядные приложения со встроенной поддержкой стандартов COM/DCOM, ActiveX, улучшенными средствами работы с базами данных и т.д. Варианты поставки, судя по всему, будут аналогичны второй версии.


    5. Сколько дискового пространства, памяти, и т.д., нужно для работы Delphi?


    Минимальная установка Delphi 1.0 требует приблизительно 30 Мбайт на диске, и полная установки - 80 Мбайт. Чтобы Delphi работал хорошо, нужен 486 процессор с 8 Мбайт ОЗУ, хотя мы рекомендовали бы 16 Мбайт. Практика показывает, что скорость CPU не является критическим параметром.


    Для 32-разрядных версий Delphi требования увеличиваются. Полная установка занимает чуть более 100 Мбайт, оперативной памяти желательно иметь не менее 16 Мбайт. 32 Мбайт ОЗУ достаточно, чтобы комфортно работать и отлаживать программы в Delphi, используя при этом загруженный на этой же машине Local Interbase.


    6. Сколько занимает места программа, выводящая текст "Hello, World!", изготовленная на Delphi?

    Меньше 170Кб, если имеется в виду программа, "собранная" обычным для Delphi способом. При этом сразу же подключается стандартная поддержка форм и пр. Тем не менее при помощи Паскаля, поддержимаемого Delphi, можно  написать программу, которая ничем по своим качествам не будет отличаться от программы, написанной в BP7.0. (То есть размер такой программы может быть около 15Кб) Рекорд — 3.5 Кб!!


    Но и это еще не все. Концепция пэкиджей (paсkages), введенная в Delphi 3.0, позволяет создавать программы, код которых будет составлять всего несколько килобайт + разделяемые между всеми приложениями библиотеки времени выполнения, оформленные в виде DLL.


    7. Hасколько трудно научиться работе с Delphi?

    Если вам повезло, то вы уже имеете большой опыт работы и с Borland Pascal With Objects, и с Visual Basic. Если вам подходит данное описание, тогда Delphi будет для вас сразу понятен. А теперь для остальных: Чтобы полностью  использовать возможности среды Delphi, вы должны знать Pascal , вы должны иметь некоторые знания об объектно-ориентированном программировании и вы должны знать о программировании событий. Если вы преодолели эти три препятствия, то вы имеете все необходимые знания. С другой стороны , большинству людей не нужно полностью использовать все возможности среды. Если вы хотите создать приложение, которое не делает ничего особенно причудливого, то даже без особого программирования вы в течение 5-10 минут сможете собрать из визуальных компонент Delphi что-нибудь достаточно мощное и на удивление работоспособное.

    Приложения, созданные в среде Delphi, отличаютя повышенной надежностью. Встроенные механизмы RTTI и обработки исключений вместе со строго типизированным языком Object Pascal изначально закладывают в программы устойчивость к всевозможным сбоям, которые могут произойти в операционной системе, на SQL-сервере или непосредственно в вашей программе. Даже ничего не делая самому, всегда можно как минимум узнать, что произошла ошибка и где, а зачастую и получить исчерпывающую дополнительную информацию. Во многом это заслуга и компилятора Borland с языка Pascal, история которого насчитывает более 13 лет.


    8. Можно ли создавать многопользовательские приложения для баз данных в Delphi Desktop или Developer?

    Используя Delphi Desktop 1.0 или Delphi Developer 2.0, можно разработать приложение, которое общается с каким-нибудь SQL сервером, используя ODBC драйвер. Hе было особых проблем, чтобы заставить работать ODBC, и скорость доступа к данным была вполне приемлема. Тем не менее те, кто уже успел опробовать оба варианта, в один голос утверждают, что через IDAPI работать быстрее. В Delphi Desktop 2.0 работа через ODBC не поддерживается.


    9. Какова история появления Delphi?

    Delphi — потомок Турбо-Паскаля , который впервые появился в 1983г для операционной системы CP/M . Турбо-Паскаль был перенесен в MS-DOS в начале 1984г. Hа протяжении всего начала истории IBM PC, Турбо-Паскаль был действительно наиболее популярным языком для серьезных разработок - главным образом потому, что это был настоящий компилятор, включающий редактор текстов программ и все необходимое, что стоило $19.95 и выполнялось на компьютере со 128 Кбайт. Borland представил Турбо-Паскаль для Windows в 1990г. Последней версией Borland Pascal (как это стало называться), не считая Delphi, был версии 7.0 в конце 1992г.


    Delphi 1.0 разрабатывался что-то около 18 месяцев или двух лет. Выпускались различные beta-версии и пререлизы, включая несколько сотен копий, которые отдали во время выставки Software Development '95. Delphi официально анонсирован в США 14 февраля 1995г, и первые копии были разосланы 28 февраля. В России Delphi появился в первых числах апреля, хотя ранее действовала программа раннего ознакомления (EEP) в рамках которой те, кто спешил начать осваивать новый продукт, получали бета-версию Delphi и пререлиз документации.


    После выпуска первой версии продукта компания Borland направила свои усилия на перенос Delphi в 32-разрядные операционные системы - Windows 95 и Windows NT. Этот процесс успешно завершился и 10 февраля 1996г. Borland Intrenational объявил о выходе второй версии продукта, которая генерирует уже 32-разрядный код для Windows 95 и Windows NT и использует все преимущества 32-разрядных приложений - более высокая скорость обработки данных, большее количество возможностей для приложения и др.


    На данный момент ожидается выход третьей версии Delphi, окончательный которой предположительно появится в конце весны-начале лета 1997г. На основании предварительных версий можно сказать, что Delphi 3.0 - это еще более мощный продукт, чем Delphi 2.0, который позволяет использовать в своих разработках все основные стандарты, имеющиеся на платформе Win32/Intel - ActiveX, COM/DCOM, ISAPI/NSAPI и т.д.


    10. Где можно приобрести копию Delphi?

    У любого дилера Borland, а сейчас Delphi Desktop продаются, кажется, даже в книжных магазинах. Тем не менее перед приобретением Delphi поинтересуйтесь, какую поддержку обеспечивает дилер. Delphi - это довольно большой продукт, и его поддержка всегда будет нелишней.


    11. Приведите сравнение производительности IDAPI и других аналогичных продуктов.

    Демо-центр такими исследованиями не занимался, в них не было ни интереса ни необходимости. Однако кто-то такие тесты проводил, например по данным Jin Mai (mwj@pipeline.com) на одной таблице с 60 тыс. записей производительность операции SELECT следующая[1]:

    Инструмент   Формат базы данных  Время в секундах 
    Delphi  Paradox  22
    VB  Access  60
    Delphi  ODBC to Access  30 
    Access  ODBC to local Watcom  48 
    VB  ODBC to local Watcom  33 

    Hесмотря на весьма странные данные во второй строке (VB/Access),  автор тестов их подтверждает.


    12. Как можно войти в контакт с представителями Borland?

    Если вы имеете соответствующие возможности, вы можете получать информацию прямо из Borland:

    www.borland.com

    • anonymous to ftp.borland.com.  

    Телефоны Borland АО — российского представительства Borland International:

    (095)-366-4298

    • (095)-366-3973  

    Вы можете поговорить с представителями Демо-центра по клиент-серверным технологиям Borland (компания Epsylon Technologies) по телефонам:

    (095)-913-5608

    • (095)-913-2934

    • (095)-459-1333

    • (095)-535-0319

    • (095)-535-5349

    Можете отправить письмо с вопросами по адресу:

    delphi@demo.ru.


    13. Какую техническую поддержку может получить покупатель Delphi?

    Можете звонить напрямую в Borland International, все необходимые адреса есть в сопроводительных документах в коробке с Delphi.

    Можете направить письмо по электронной почте в Borland International по адресу techinfo@borland.com, однако следует отметить, что качество поддержки в большой степени зависит от дилера.

    Hачала свою деятельность Ассоциация пользователей Delphi. В Борланд АО организованы семинары (пока бесплатные). Раз в две недели по средам с 14:00 до 16:00 проходит тематический семинар о Delphi и других продуктах Borland. Его проводят практические специалисты по Delphi из различных организаций, в том числе и из демо-центра. Даже если вы издалека, имеет смысл посетить семинар — наверняка вы услышите что-либо интересное для себя. Тематика семинаров — самая разнообразная.

    Для того, чтобы попасть на семинар, надо позвонить в Демо-центр по телефону: 913-5608, и записаться на семинар.

    Методические материалы для проведения таких семинаров сохраняются, так что тем, кто желает у себя в регионе организовать нечто подобное, звоните в Демо-центр Александру Сергееву.


    14. Выйдет ли версия Delphi, где в качестве компилятора будет использоваться BC++?

    Вопрос устарел. Такой продукт (Borland C++Builder) вышел 4 февраля 1997г. По своим возможностям он практически равноценен Dehlpi 2.0. Получить более подробную информацию о данном продукте вы можете получить на страничках нашего сервера, посвященных C++Builder.


    15. Сколько стоит Delphi в России?

    Полная стоимость текущих версий продукта составляет:

    Delphi 1.02 Client/Server 
    Delphi 1.02 for Windows (Desktop)  $
    Delphi 2.01 Client/Server Suite  $
    Delphi 2.0 Developer  $
    Delphi 2.0 Desktop  $        

    Цены на Delphi 3.0 не объявлены (продукта пока нет), но предполагается, что они будут на уровне Delphi 2.0. Полный список цен на продукты, а также всевозможных вариантов Upgrade приведен в нашем прайс-листе.


    16. Что можно, и чего нельзя разработать в среде Delphi? В частности, можно ли разработать в Delphi внутреннюю структуру данных для SQL-cервера, хранимые процедуры и пр.?

    Понятно, что для такого открытого инструмента, как Delphi, границы "дозволенного" весьма условны. В частности, хороший пример — появление в более поздних версиях визуального компонента для работы с хранимыми процедурами. То есть это было нельзя для FieldTest 4 и стало возможным для пререлиза.

    Второй пример, который появился только с версии 1.15 Delphi FAQ. В стандартной поставке Delphi не существует визуального компонента, поддерживающего обработку event alerts (событий) в InterBase. Тем не менее оказалось, что изготовить такой компонент не так уж и трудно. В Delphi 2.0 есть пример аналогичного компонента, который на самом деле оказался весьма кривым и потребовал существенной доработки для возможности нормального использования.

    Вышли компоненты, позволяющие проводить процедуру инсталляции разработанного вами приложения, проводить сетевой обмен, обмениваться данными через cc:mail или Lotus Notes.

    Появились компоненты, реализующие прямые линки к BTrieve, AS/400, мэйнфреймам, Informix.


    17. Чем можно воспользоваться для создания инсталляционных версий приложений, разработанных в Delphi?

    Для Delphi 1.0 можно, например, попробовать:

    1. Wise. Он неплохо подходит и для VB, и для BC++.

    2. Kurt. Создает иконы, меняет INI, распаковывает сжатые файлы.

    3. Существует несколько компонентов, которые помогают в создании инсталляторов и Shareware-версий продукта. Однако, последнее время практически стандартом стал инсталлятор InstallShield. Его 32-разрядный вариант для Delphi (InstallShield Express for Delphi) входит в стандартную поставку всех версий Delphi 2.0.


    18. Должен ли я знать все относительно Windows API, чтобы использовать Delphi?

    Может возникнуть чувство, что вы должны знать относительно Windows API в Delphi больше, нежели в Visual Basic. Это не так; вы можете работать в обеих средах с минимальным пониманием внутренней организации Windows. Однако, в обоих случаях, вы должны знать, по крайней мере, кое-что относительно Windows API, чтобы "выжать" максимум из того, что у вас есть. Различие в том, что Delphi предоставляет вам гораздо больше возможностей, чтобы делать все эти интересные вещи.


    19. Должен ли я знать объектно-ориентированное программирование, чтобы использовать Delphi?

    Хорошо бы. Инструментальные средства проектирования интерфейса пользователя Delphi производят объектно-ориентированный код. Однако, если вы знакомы с Visual Basic или Powerbuilder, вы, вероятно, имеете достаточное понимание OOP (Object Oriented Programming). Вы можете сделать многое в Delphi без необходимости создавать ваши собственные объекты; но при создании новых объектов действительно важно знать тонкости OOP.

    Совместимость

    1. Какие операционные системы Delphi поддерживает?

    Версия Delphi 1.0 предназначается для Windows 3.1x. Hет причин, по которым Delphi 1.0 не работал бы в системах, которые обеспечивают эмуляцию Windows 3.1, подобно OS/2 Warp, Windows NT, UnixWare 2.0 и т.д.

    Delphi 2.0 предназначена для работы под Win32/Intel - на данный момент это Windows 95 (в т.ч. OSR2), Windows NT (3.5 и 4.0).

    Delphi 3.0 также работает под Win32, но на данный момент некоторые возможности, предоставляемые продуктом можно использовать только в ОС Windows NT 4.0. Возможно, очередная версия Windows 95 (Windows 97), будет обладать этими функциями.


    2. Какие средства управления проектом совместимы с Delphi?

    Delphi 1.0 Client/Server поддерживает совместимость с PVCS компании InterSolv.

    Delphi 2.0 Client/Server Suite включает в себя комплект PVCS, интегрированный в IDE.

    Кроме того, должна быть возможность совмещения с большинством систем контроля исходных текстов. Однако, формы в Delphi сохраняются в двоичном формате, так что пакет управления исходными текстами должен поддерживать двоичные данные для того, чтобы это работало с Delphi . Если у вас есть администратор исходных текстов, который просто не может иметь дело с двоичными файлами, то согласно документам, вы может сохранять формы в виде ASCII текста для редактирования или в целях контроля за версиями. Файлы ASCII могут также загружаться и снова сохраняться в двоичном *.DFM формате.

    MKS Source Integrity (MKS SI - 2-ой на рынке администраторов исходного текста для PC/PCLAN с более чем 35,000 разработчиками, использующими SI) также предлагает средства интеграции в Delphi. Эти средства в настоящее время поставляются наряду с MKS SI.


    3. Можно ли в Delphi использовать DLL, разработанные в C или C ++?

    Delphi способен вызывать и получать обратные вызовы (callback) из любого стандартного модуля DLL для Windows.


    4. Можно ли вызывать код, созданный в  Delphi из C или C ++?

    Delphi может генерировать DLL, которые можно вызывать из C, C++, Visual Basic, PowerBuilder, или чего-нибудь еще, что понимает стандартные Windows DLL. Имеется пример DLL в каталоге DEMOS\DB\DLL из комплекта Delphi.


    5. Что известно о пакетах других фирм, которые работают (или не работают) с Delphi?


    6. Есть ли поддержка Dynamic Data Exchange (DDE), VBX, OLE, OCX, OpenDoc?

    Да, Delphi позволяет с разной степенью комфорта использовать все технологические стандарты, имеющиеся в Windows.


    7. Можно ли на Delphi написать VBX или OCX?

    Можно. Ведь по сути дела VBX — это .dll, написанная по определенным правилам. Однако скорее всего вам потребуется дополнительно соответствующий SDK от компании Microsoft. Существует даже статья, где обсуждаются все аспекты такой работы.

    Относительно OCX все то же самое, что и относительно VBX.

    Delphi 3.0 позволяет визуально создавать элементы ActiveX.


    8. Какие генераторы отчетов можно использовать с Delphi кроме ReportSmith?

    В порядке предпочтения:

    1. Имеется QuickReport — генератор отчетов, сделанный в виде компонентов и встраиваемый непосредственно в приложение. Успех 16-разрядной версии привел к тому, что начиная с версии Delphi 2.0 32-разрядный QuickReport входит в стандартный вариант поставки.

    2. Crystal Reports — подключается через VBX.

    3. R&R Report Writer — хорошо работает с Delphi, позволяет выполнять запросы, preview, вызовы DLL, передачу параметров. (производитель — Concentric Data Systems)

    4. Также есть еще несколько генераторов отчетов, выполненных аналогично QuickReport в виде компонентов.

    Следует также заметить, что на данный момент Borland International продал ReportSmith другой фирме, так что его присутствие в Delphi 3.0 под вопросом.

    Базы данных - Interbase и локальные данные

    1. Каково определение IDAPI? Что такое SQL Links?

    IDAPI это Integrated Database Application Program Interface. BDE (Borland Database Engine) — средство доступа ко множеству источников данных через один API. IDAPI - это просто API для BDE. IDAPI включает все функции, необходимые для доступа к данным, манипулирования ими и т.д. Delphi, Borland C++, C++Builder, Intrabulder, dBASE for Windows, и Paradox for Windows используют эти функции. Вы можете использовать их в своих программах. Вы получите документацию, если купите BDE. Там перечислены все доступные функции и что они делают. Если посмотреть на исходники VCL в Delphi, то можно увидеть, как они используются. Они начинаются с "Dbi" (e.g. DbiCreateTable).

    SQL Links — набор родных драйверов (native drivers), которые нужны для работы с удаленными серверами баз данных.


    2. Hеобходим ли IDAPI для доступа к данным в Delphi? Можно ли включить IDAPI внутрь EXE, чтобы распространять программу без установки IDAPI на пользовательском компьютере?

    IDAPI необходим для доступа к данным в Delphi. Сам IDAPI во внутрь исполняемого файла ни коим образом не встраивается (да это было бы и не целесообразно). Вместе с Delphi поставляется редистрибутивный вариант BDE, которая устанавливает на "чистую" машину лишь только BDE. может создавать инсталляторы, которые устанавливают на машину как ваше приложение, так и BDE.

    InstallShield Express for Delphi

    Delphi 3.0 включает средства для создания "тонких" клиентов, работающих на машине, на которой не установлен BDE. Подробности смотрите в FAQ по Delphi 3.0.


    3. Где можно найти описание функций и типов данных BDE?

    DBIPROCS.INT в директории DELPHI\DOC\ содержит список функций BDE, передаваемые параметры, возвращаемое значение и краткое описание каждой. DBITYPES.INT — список типов, используемых функциями BDE. Для вызова любой функции BDE добавьте следующие модули в раздел uses: DBITYPES, DBIPROCS и DBIERRS.

    Delphi 2.0 включает в себе описание функций BDE в формате WinHelp. Также все три модуля из Delphi 1.0 (DBITYPES.DCU, DBIPROCS.DCU и DBIERRS.DCU) теперь объединены в единый BDE.DCU.


    4. Можно ли программным образом добавить псевдоним (alias) в IDAPI.CFG?

    В BDE есть для этого функция DbiAddAlias.

    В Delphi 2.0 данная функциональность находится внутри компонента TSession (методы AddAlias, AddStandardAlias).


    5. Я получаю сообщение от BDE при редактировании записи 'Multiple records found but only one expected'. Что бы это значило?

    Вам может потребоваться создать уникальный индекс в таблице для того, чтобы каждую строку в таблице можно было однозначно идентифицировать. Например, попытайтесь изменить структуру таблицы и добавить поле для заполнения его уникальными значениями.


    6. Обработка исключений (exceptions) BDE.

    Информация об ошибке BDE может быть получена для использования в приложении из объекта EDBEngineError. Исключительная ситуация EDBEngineError обрабатывается в программе с помощью конструкции try ... except. Когда возникает исключительная ситуация BDE, то может быть создан объект EDBEngineError и различные поля этого объекта могут быть использованы для программного определения, что не в порядке и что требуется для исправления ситуации. Далее, для данной исключительной ситуации может быть сгенерировано несколько сообщений об ошибках. Это требует организации перебора сообщений об ошибках для получения нужной информации.


    7. Какому стандарту SQL соответствует SQL в InterBase?

    SQL в Local & Remote InterBase соответствует SQL-92 с элементам SQL III (более поздними расширениями).


    8. Как удалить генераторы (GENERATORS) из базы данных InterBase?

    Никак. К сожалению, в существующих версиях InterBase это невозможно.


    9. Как выбрать протокол при соединении с InterBase из Delphi?

    В Server Manager (ibmgr.exe), Windows interactive SQL (wisql.exe) и Communication Diagnostic Tool (comdiag.exe) Вы отдельно задаете имя сервера, протокол и 'путь на базу' (локальный путь на сервере, а не путь до базы с Вашей машины)

    А в BDE Configuration Utility все немного не так — на странице Drivers у драйвера Interbase есть параметр SERVER NAME, заполненный как IB_SERVER:/PATH/DATABASE.GDB, а у любого InterBase Alias есть параметры SERVER NAME и PATH, но ни у драйвера ни у алиаса нет протокола. Для Interbase протокол указывается стилем написания пути к базе:  

    Протокол  SERVER NAME  Пример
    TCP/IP  IB_SERVER:PATH\DATABASE.GDB  nt:c:\ib\base.gdb; unix:/ib/base.gdb
    IPX/SPX  IB_SERVER@PATH\DATABASE.GDB  nw@sys:ib\base.gdb
     NETBEUI \\IB_SERVER\PATH\DATABASE.GDB \\nt\c:\ib\base.gdb 

    10. Можно ли использовать какие-нибудь ODBC драйверы, которые получены с другими СУБД?

    В общем, да. Мы не столкнулись с какими-то ODBC драйверами, которые не работают с Delphi, но, с другой стороны, мы не пробовали действительно экзотические драйверы. Основные трудности вызвали специфические ODBC-драйверы от Microsoft. Похоже, что ODBC и ODBC от Microsoft - это разные вещи.


    11. Проблемы с именами таблиц в ODBC-драйверах.

    При использовании ODBC-драйверов, если у вас появляются ошибки при открытии таблицы - установите порядок сортировки (SORT ORDER) хотя-бы Paradox ASCIIi. Это, например, помогает при доступе к базам данных Lotus Notes (ODBC-драйвер фирмы Casahl).


    12. Какой формат данных предпочесть в Delphi? dBase или Paradox?

    Если вам действительно все равно, то вот несколько пунктов 'за' формат Paradox:

    1. Широкий выбор типов полей, включая автоинкремент, BLOBs, и т.п.

    2. Соблюдение целостности данных, контроля данных, обновления индексов на уровне ядра BDE.

    3. Первичный индекс таблицы автоматически соблюдает уникальность записей, вторичные индексы обеспечивают отсортированный "вид" на записи таблицы.


    13. Как нужно писать функцию "change password"  для таблицы Парадокса? 

    Нет способа сделать это в пределах Delphi VCL. Кажется, это довольно серьезное упущение. Однако, есть возможность сделать это напрямую через Borland Database Engine через интерфейс предоставляемый модулями DBIPROCS.DCU и DBITYPES.DCU () или BDE.DCU (). Нужно использовать функцию DbiDoRestructure.


    14. Есть ли какая-нибудь процедура для перестройки разрушенного индекса, типа TUTILITY.EXE из PdoxWin?

    BDE включает функцию для этого — DbiRegenIndexes.


    15. Есть ли какая-нибудь процедура для упаковки таблицы dBase?

    В BDE есть функция DbiPackTable.


    16. Как для .dbf таблицы создать индекс по выражению?

    Нужно использовать процедуру AddIndex с параметром ixExpression, например:

    Table1.AddIndex('NewIndex','Field1 * Field2 + Field3', [ixExpression]);


    17. Как создать в Paradox вторичный индекс с упорядочиванием по убыванию?

    Используйте флаг ixDescending:

    Table1.AddIndex('NewIndex', 'CustNo;CustName', [ixDescending]);


    18. Хочу узнать номер текущей записи, как это сделать?

    В общем случае — никак. В случае таблицы Paradox — есть в BDE функция DbiGetSeqNo, которая возвращает логический номер записи. Но при использовании на форме TDBGrid она может давать не всегда правильные значения.


    19. Как посмотреть удаленные записи в таблице .dbf? А как их восстановить?

    Для того, чтобы удаленные записи были доступны, нужно установить соответствующее свойство для курсора с помощью функции BDE DbiSetProp:

    DbiSetProp(hObj(Table1.Handle), curSoftDeleteOn, 1);

    Проверка удалена запись или нет производится через функцию чтения записи DbiGetRecord. Для восстановления записи применяется функция DbiUndeleteRecord.


    20. Упаковка таблицы.

    Упаковать таблицу DBF можно открыв ее компонентом TTable и вызвав функцию BDE DbiPackTable:

    Result := DbiPackTable(Table1.DbHandle, Table1.Handle, nil, szDBase, True);

    21. Почему я получаю ошибку 'Index out of range' когда использую TTable.FindNearest и TTable.FindKey для таблицы dBase с индексом по выражению?

    Методы TTable.FindKey и TTable.FindNearest не могут работать с таким видом индексов. Вместо этих методов используйте TTable.GotoKey и TTable.GotoNearest, которые прекрасно работают с ними.


    22. Как программным образом создать таблицу Paradox с автоинкрементным полем?

    Вам следует использовать компонент TQuery и SQL-предложение типа:

    CREATE TABLE "PDoxTbl.db" (ID AUTOINC, Name CHAR(255), PRIMARY KEY(ID));


    23. Почему я не могу использовать опцию ixUnique при создании индекса в таблице Paradox с помощью метода AddIndex компонента TTable?

    Опции, используемые в методе AddIndex компонента TTable зависят от типа таблиц. Например, опция ixUnique работает с таблицей dBase, но не с Paradox. Следующая таблица показывает, как эти опции используются для таблиц dBase и Paradox.  

    Index Options  dBase  Paradox
    ixUnique  *
    ixExpression  *
    ixDescending  *
    ixNonMaintained  *
    ixPrimary  *
    ixCaseInsensitive 


    24. Генерация уникальных идентификаторов для таблиц.

    Для более полного ознакомления с этим вопросом рекомендуется обратится к статье Максима Михеенкова в 1-ом номере российского журнала СУБД за 1995 год. А коротко можно сказать следующее.

    Для таблиц Paradox вы можете пользоваться специальными типами полей, гарантирующими уникальность значения поля в записи — типы AutoIncrement и TimeStamp (с использованием функций DateXXX — модуль SysUtils).

    Для Interbase вы можете использовать генераторы, которые возвращают уникальное значение.

    Если же такое решение вам не подходит, то можно перед помещением записи в таблицу отыскивать максимальное уже имеющееся в таблице значение этого поля и заполнять поле в записи значением, на единицу больше. Для разграничения пользователей при одновременном доступе можно хранить это значение в отдельной таблице и открывать ее эксклюзивном режиме.


    25. Можно ли использовать Crystal Report с таблицами формата Paradox 5.0?

    Да, можно, только если вам удастся получить ODBC-драйвер для таблиц Paradox. Во всяком случае на Crystal BBS находится файл bde.zip, который и содержит этот драйвер.


    26. Как открыть таблицу dBase, у которой поврежден (утерян) индексный MDX файл?

    Как я могу использовать таблицу dBase без необходимого для нее MDX файла?

    При создании таблицы dBASE с индексным файлом MDX в заголовке DBF файла устанавливается байт со смещением 28 (десятичное) от начала файла. При открытии таблицы, у которой данный байт установлен, также происходит попытка открыть MDX файл; если это не удается, то возникает исключение (exception). Для решения проблемы достаточно прописать в DBF файл 0 по указанному смещению.


    27. Как определить номер текущей записи для набора данных?

    Если набор данных основан на таблицах Paradox или dBASE, то номер записи можно определить с помощью вызовов BDE. BDE не поддерживает номер записи для наборов данных на SQL сервере; если ваш сервер поддерживает нумерацию записей, вам нужно обратиться к его документации.

    Базы данных — прочие SQL сервера

    1. Как осуществляется доступ к базе данных из Delphi?

    Сначала вы должны в утилите BDE Configuration Utility (bdecfg или bdecfg32) определить псевдоним для базы данных, с которой вы хотите работать. Это позволит вам избежать написания сложного пути к базе данных в вашем приложении; вы теперь только ссылаетесь на псевдоним. Затем вы создаете минимум три объекта на форме: TTable или TQuery, который фактически общается с базой данных через псевдоним и получает данные; объект TDataSource, который связывает данные и визуальные компоненты; по крайней мере один компонент, отображающий данные.

    Если вы потратили несколько часов, чтобы заставить все это работать, но ничего не получилось, попробуйте установить свойство Active у TTable или TQuery в True. Это откроет таблицу в базе данных. 


    2. Какие серверы данных поддерживает Delphi?

    Delphi (в старших вариантах поставки) напрямую работает с Oracle, Informix, InterBase, DB/2, Sybase, MS SQL Server. Вы можете использовать ODBC драйверы третьих фирм, чтобы работать с любым сервером. Наиболее известные производители качественных ODBC драйверов:

    • InterSolv (www.intersolv.com)

    • Visigenic Software (www.visigenic.com)

    Есть специализированный вариант Delphi 2.0 — Delphi/400, который ориентирован исключительно на работу с AS/400.


    3. Cуществует ли способ работать из Delphi с AS/400, исключая ODBC?

    В состав Delphi 1.0 не входит, но доступен отдельно визуальный 16-разрядный компонент компании Gerald Limited.

    Есть специализированный вариант Delphi 2.0 - Delphi/400, который ориентирован исключительно на работу с AS/400.


    4. Cуществует ли способ работать из Delphi с Lotus Notes (IBM Notes), за исключением ODBC?

    Существует только 16-разрядный линк для Notes. В состав Delphi он не входит (разработан компанией Brainstorm) и поставляется отдельно по каналам Borland. Спрашивайте дилеров Borland.


    5. Можно ли создавать с помощью Delphi сетевые приложения не для схемы клиент-сервер, а для схемы с разделяемыми файлами (как, например, в FoxPro)?

    Компонент TTable имеет свойство Exclusive; если Exclusive=False, то одну и ту же таблицу могут просматривать и редактировать несколько пользователей. При редактировании таблицы текущая запись автоматически блокируется. Если есть необходимость заблокировать несколько записей или всю таблицу, то для этого придется использовать функции BDE — модуль DBIPROCS (

    ) или BDE (

    ). Кстати, существенно, что это можно делать и на Delphi Desktop - отсюда вывод: сетевые приложения (в файл-серверной архитектуре) можно делать и на Delphi Desktop.


    6. Можно ли работать при помощи Delphi не с IDAPI — или ODBC-драйверами, а с "родными" API каких-либо СУБД?

    Да, можно. Это еще один пример открытости Delphi. В одном из проектов Демо-центра появилась необходимость прямого обращения к API SQL-сервера. Был написан соответствующий интерфейс поверх Borland InterBase API - и все заработало. Аналогично: существуют написанные компоненты для работы с AS/400 и для мэйнфреймов.


    7. Какие версии Informix (Online, I-NET) поддерживают SQL Links?

    BDE версии до 2.51 поддерживает работу с клиентской частью Informix ESQL/C I*NET 4.x. BDE версии 2.51-2.52 поддерживается ESQL/C I*NET 5.x.

    32 SQL Link версии 3.5 и выше поддерживает ESQL/C I*NET 7.x.


    8. Как работать с таблицами в виде текстовых файлов (ASCII)?

    Информацию об этом вы можете найти в каталоге DELPHI\DOC. Файл называется ASCIIDRV.TXT.


    9. Как правильно указать имя сервера Oracle?

    Пишите имя по правилам Oracle - перед именем не забудьте поставить @.


    10. Что такое транзакции (Transactions)?

    SQL database серверы обрабатывают запросы в 'логических единицах работы' которые и называются транзакциями. То есть транзакция - это группа связанных операций (SQL запросов) которые все должны быть выполнены успешно перед тем, как сервер закончит (commit) изменение базы данных. Либо вся это группа будет выполнена, либо нет. Транзакции обеспечивают целостность базы данных ...


    11. Как в Delphi управлять транзакциями?

    В Delphi может управлять транзакции так:

    1. Implicitly — сам стартует и коммитит транзакции по необходимости, когда программа вызывает метод Post.

    2. Explicitly —

     1. StartTransaction, Commit & RollBack методы TDatabase.

     2. При помощи SQL запросов через TQuery — это зависит от конкретного SQL сервера.

    Неявные (1) транзакции выполняют TTable & TQuery.

    Для явных (2.1) транзакций требуется TDatabase.

    Для явных (2.2) транзакций требуется TQuery.

    Важно:

    При определении параметров драйвера SQLPASSTHRU MODE - определяет будут ли passthrough SQL (PSQL) & стандартные вызовы BDE (BDEC) использовать один и тот же connect к SQL серверу.

    Возможные варианты:

    1. NOT SHARED — PSQL & BDEC используют разные соединения с базой и следовательно влияют (интерферируют) друг на друга также как разные пользователи одной базы (см. Transaction isolation levels)

    2. SHARED:

     1. SHARED AUTOCOMMIT — PSQL & BDEC используют одно соединение с базой, каждый PSQL запрос автоматически коммитятся.

     2. SHARED NOAUTOCOMMIT — PSQL & BDEC используют одно соединение с базой, PSQL запросы коммитятся 'вручную' (способом 2.1).

    Еще важнее:

    Если Вы все-таки решили управлять транзакциями при помощи SQL запросов через TQuery, то SQLPASSTHRU MODE должно быть NOT SHARED, иначе Implicit & Explicit транзакции могут влиять друг на друга и привести к 'неожиданным результатам'!


    12. Использование формата Access 2.0 в Delphi.

    Подробное описание подключения:

    • Откройте Windows Control Panel, откройте икону ODBC.

    • Добавьте драйвер Access в список доступных, если он не указан в списке текущих драйверов.

    • Укажите какое-либо имя в "Data Source Name".

    • Выберите файл БД кнопкой Select Database.

    • Если вы хотите создать новый — выполните пункт Create Database  

    • Запустите Database Engine Configuration

    • Нажмите кнопку New ODBC Driver

    • Выберите драйвер типа ACCESS DATA

    • В качестве Default Data Source Name выберите предложенное.

    • Создайте новый псевдоним в разделе Alias

    • Укажите тип драйвера, путь и имя к нужному файлу БД Access.

    • Сохраните изменения и закройте Database Engine Configuration

    Примечание: избегайте русских названий имен таблиц, использования пробела в именах таблиц, а также русских имен полей в DB Access. 


    13. Можно ли при помощи Delphi реализовывать проекты, не имеющие отношения к базам данных? Если да, то имеет ли это смысл?

    В таком случае стоит воспользоваться вариантом Delphi Desktop или Developer. Выигрыш по скорости разработки очевидно будет значительным; в частности, разработка интерфейса программы под Windows производится действительно скоростными методами. Хороший пример — Screen Saver для Windows.

    Общие вопросы по Delphi и данному FAQ (часть 2)

    Базы данных — компоненты и VCL.

    1. Какие визуальные компоненты для работы с данными входят в  Delphi?

    Различные версии Delphi содержат следующие наборы компонентов:

    Компоненты Краткое описание Версии Delphi
    TDBGrid  Представление данных в виде таблицы (очень сложный и наиболее популярный компонент) 1 2 3
    TDBEdit  Редактирования одного поля 1 2 3
    TDBNavigator  Как видно из названия, компонент позволяет перемещаться по таблице 1 2 3
    TDBLabel  Статическое отображение содержимого поля 1 2 3
    TDBMemo Редактирования текста в поле типа BLOB 1 2 3
    TDBImage Отображение картинок из BLOB-а 1 2 3
    TDBRadioGroup, TDBCheckBox Дополнительные средства отображения данных 1 2 3
    TDBComboBox, TDBListBox Упрощают ввод данных, предлагая несколько заранее определенных вариантов 1 2 3
    TDBLookupListBox, TDBLookupComboBox То же самое, но возможные варианты выбираются из другой таблички 1 2 3
    TDBCtrlGrid Вариант представления записей с произвольным расположением полей 2 3
    TDBChart  Компонент для построение графиков и диаграмм на основании данных, хранящихся в таблице 3
    TDecisionGrid, TDecisionChart Компоненты для поддержки принятия решений 3

    2. Использование псевдонимов в запросе SQL.

    Я делаю запрос по двум таблицам разных форматов, находящихся по разным псевдонимам.

    SELECT DB1.Column1, DB2.Column2 FROM :Alias1:DB1, :Alias2:DB2

    но в результате получаю ошибку 'неизвестный тип поля "Alias1:DB1"'

    На самом деле вы получаете ошибку Unknown Keyword, следовательно всего-лишь нужно заключить псевдоним и имя таблицы в двойные кавычки.

    SELECT D1.Column1, D2.Column2 FROM ":Alias1:DB1" D1, ":Alias2:DB2" D2

    Вообще экспериментировать с SQL-запросами проще следующим образом — создайте запрос QBE, настройте его так, как вам нужно, а затем оттранслируйте его в SQL. В результате вы получите правильный текст нужного вам SQL-запроса. Владельцы Delphi Client/Server могут использовать также и Visual Query Builder. Однако, не все QBE-запросы могут быть оттранслированы в SQL.


    3. Ошибка в SQL запросе.

    У меня есть TQuery и TDataSource. В свойстве SQL для TQuery я пишу

    SELECT * FROM dbo.AnyTable

    база данных на MS SQL Server. Когда я устанавливаю Active в True, то получаю ошибку:'Token not found. Token :dbo. line number:1'. Что не так?

    Если свойство RequestLive=True, то имя таблицы нужно брать в кавычки:

    SELECT * FROM "dbo.table"

    Если свойство RequestLive=False, кавычек не требуется:

    SELECT * FROM dbo.table


    4. Проблемы при работе с MS Access через TQuery.

    Я безуспешно пытался использовать данные из Microsoft Access иначе, нежели просто с помощью TTable.  Используя TQuery я могу только читать результат, но не могу редактировать. После "login screen" возникает сообщение типа 'Passthrough SQL connection must be shared'.

    Измените в настройке псевдонима (alias) пункт 'SQLPASSTHRU MODE' на 'SHARED AUTOCOMMIT'.


    5. Как создать таблицу при помощи SQL (или почему не работает TQuery.Open)?

    TQuery.Open возвращает результат в виде курсора, в связи с этим он работает только для тех выражений, которые возвращают курсор. CREATE TABLE возвращает только результат операции - поэтому для выполнения этого выражения необходимо использовать TQuery.ExecSQL. Но и это может не сработать, если конкретный драйвер БД не поддерживает операцию создания таблиц - для получения характеристик драйвера используйте функции BDE (DbiOpenDriverList, DbiGetDriverDesc). 

    6. Возврат значения select max() и подобных SQL-выражений.

    Я хочу выполнить SQL-выражение и получить результат в свою переменную, что-то типа

    SELECT MAV(FieldA) FROM TableB INTO :VariableC;

    Вам не нужно использовать оператор INTO для программного доступа к результату — его можно получить используя свойства Fields или FieldByName соответствующего компонента TQuery.

    TQuery1.Add(' SELECT MAX(FiledA) FROM TableB ');

    TQuery.Open;

    ...

    VariableC := TQuery1.Fields[0];

    Или, если результат нужно визуально отобразить, достаточно подключить к используемому TQuery компоненты TDataSource и TDBText.


    7. Автоматический подсчет сумм при помощи TQuery.

    Меня интересует возможность подсчета суммы по таблицам, которые уже находятся на форме.

    Есть очень простой способ — предположим, что у вас есть на форме Query1, DataSource1, DBGrid1. Добавьте на эту же форму компоненты Query2, DataSource2, DBText1. Установите property Query2.DataSource=DataSource1. В Query2.SQL напишите

    SELECT SUM(FieldName) FROM TableName

    где TableName — имя той же таблицы что и у Query1, а FieldName — имя столбца по которому производится подсуммирование. Далее свяжите между собой Query2, DataSource2 и DBText1.

    При изменении Query1 (если конечно Query1.RequestLive=True) Query2 будет автоматически перевыполняться. Это решение хоть и простое, но неэкономичное — особенно при большом количестве записей в исходной таблице. Более того, запрос Query2 должен иметь WHERE идентичный Query1.

    Для подсчета сумм правильнее использовать событие TQuery.OnCalcFields. Хорошим примером является X:\DELPHI\DEMOS\DB\MASTAPP\MASTAPP.DPR. 


    8. Использование кавычек в параметризированном запросе.

    Мой запрос получает параметр. Проблема в том, что строка параметра содержит " (двойную кавычку), которая приводит к Runtime Error.

    Вам необходимо использовать динамический SQL-запрос, иначе при указании например

    WHERE TABLE.FIELD = 'let"ter'

    вы получите ошибку.


    9. Как создать отдельный компонент TTable?

    Легко и просто — точно также как и обычный компонент. При этом в качестве параметра конструктору можно передавать значение nil.  

    var 

      MyTable: TTable; 

    begin 

      MyTable := TTable.Create(nil); 

      try 

        MyTable.DatabaseName := 'MyDB'

        MyTable.TableName := 'MyTable.db'

        Mytable.IndexName := 'MyIndex'

        MyTable.Open; 

        { делать то, что надо } 

      finally 

        MyTable.Free; 

      end

    end;


    10. Как узнать, какая ячейка при просмотре TDBGrid текущая?

    Здесь процедура для сохранения текущего номера строки и колонки. Следующий код в методе MyDBGridDrawDataCell обновляет переменные Col и Row (которые не должны быть локальными для этого метода) каждый раз, когда таблица перерисовывается. Используя этот код, вы можете считать, что Col и Row указывают на текущую колонку и строку соответственно.  

    var 

      Col, Row: Integer; 

    procedure TForm1.MyDBGridDrawDataCell(Sender: TObject; const Rect: TRect; 

      Field: TField; State: TGridDrawState); 

    var 

      RowHeight: Integer; 

    begin 

      if gdFocused in State then 

      begin 

        RowHeight := Rect.Bottom - Rect.Top; 

        Row := (Rect.Top div RowHeight) - 1; 

        Col := Field.Index; 

      end

    end


    11. Как выделить цветом текущую строку в TDBGrid?

    Для TDBGrid в свойстве Options установите dgRowSelect в True.


    12. Как изменить цвет ячейки в TDBGrid?

    Введите следующий код в обработчике события OnDrawDataCell:  

    procedure TForm1.DBGrid1DrawDataCell(Sender: TObject; const Rect: TRect; 

      Field: TField; State: TGridDrawState); 

    begin 

      if gdFocused in State then 

      with (Sender as TDBGrid).Canvas do 

      begin 

        Brush.Color := clRed; 

        FillRect(Rect); 

        TextOut(Rect.Left, Rect.Top, Field.AsString); 

      end

    end;

    Установите свойство DefaultDrawing в True. Здесь перерисовывается только выделенная ячейка. Если установить DefaultDrawing в False, то вы должны самостоятельно перерисовать все ячейки аналогично примеру.


    13. Как узнать, что пользователь перешел на другую запись, например, в TDBGrid?

    Переход на новую запись — это событие, которое относится не к визуальному компоненту, а к источнику данных. Соответствующее событие называется OnDataChange и имеется у компонента TDataSource.


    14. Как устанавливать собственный цвет или шрифт для столбца TDBGrid?

    Выключите property DefaultDrawing, и обрабатывайте событие OnDrawDataCell:  

    procedure TForm1.DBGrid1DrawDataCell(Sender: TObject; const Rect: TRect; 

      Field: TField; State: TGridDrawState); 

    begin 

      if Field.FieldName = 'Name' then DBGrid1.Canvas.Font.Style := [fsBold]; 

      DBGrid1.DefaultDrawDataCell(Rect, Field, State); 

    end;    

    Это приведет к тому, что содержимое столбца 'Name' будет показываться жирным шрифтом.

    В Delphi 2.0 вы можете использовать редактор столбцов для той же самой цели.


    15. Почему указатель ползунка в TDBGrid не показывает текущее положение в таблице?

    Дело в том, что TDBGrid предполагает многопользовательский доступ к таблице. В этом случае другие пользователи этой же таблицы могут добавлять или удалять записи, в результате информация о количестве записей на текущий момент становится неопределенной.

    Конечно, в однопользовательском варианте количество записей всегда известно, но поскольку TDBGrid работает через промежуточный источник данных DataSource, ему неизвестен конкретный способ доступа к данным — навигационный или SQL. Например, для SQL существует только один способ узнать количество записей — выполнить специальный запрос с их подсчетом, а на это может потребоваться значительное время.

    По всем этим причинам TDBGrid является универсальным средством для просмотра таблиц, которое работает во всех случаях и с любыми источниками данных.


    16. Как установить фокус на определенное поле в TDBGrid?

    Используйте код:

    DBGrid1.SelectedField := Table1SomeField;

    DBGrid1.SetFocus;


    17. Как создать обработчик события OnClick для TDBGrid?

    Как и всякий TControl (иерархия наследования TControl→TWinControl→TCustomControl→TCustomGrid→TCustomDBGrid→TDBGrid) у TDBGrid есть событие OnClick, но оно protected. Так что можно либо создать новый класс, производный от TDBGrid, в котором объявить это свойство как published, либо использовать другой вариант. Например, вы можете использовать событие OnColEnter.

    18. Как создать маску для TDBEdit?


    Маска относится к полю в таблице (компонент TField) а не к самому TDBEdit. Дважды щелкните мышкой на TTable и в FieldEditor'е добавьте все нужные вам поля. Когда поле выбрано в списке, его свойства показаны в Object Inspector, включая маску ввода. Связывание TDBEdit и любых других компонентов с этим TTable будет вызывать наложение маски на соответствующее поле.


    19. Хотелось бы иметь для OLE объектов, сохраненных в базе данных, компонент вроде TDBImage.

    В стандартном наборе такого компонента действительно нет. Возможно, кто-нибудь скоро напишет что-нибудь в этом роде. В принципе, можно обойтись и без данного компонента. Например, есть табличка .db с BLOB полем для OLE объекта. При движении по записям можно OLE сохранять в базе, уничтожать, создавать новый, считывать из базы.

    Чтение/запись OLE:

    • создать поток, связанный с BLOB полем

    • для OLE контейнера выполнить чтение/запись с потоком (SaveToStream и LoadFromStream)

    Естественно, OLE объект должен быть Embedded.


    20. Что нужно сделать, чтобы при открытии запароленной таблицы не появлялся диалог запроса пароля?

    Просто дайте этот пароль объекту Session перед открытием таблицы:

    Session.AddPassword('PASSWORD');

    После закрытия таблицы, пароль можно удалить RemovePassword('PASSWORD'), можно удалить все пароли: RemoveAllPasswords.

    Если ваш компонент доступа к данным (TTable или TQuery) связан с сессий, отличной от той, которая выставляется по умолчанию, то добавлять пароль нужно именно у этого компонента TSession.


    21. Как определить реальный размер поля типа BLOB, которое сохранено в таблице?

    Ниже приведена функция GetBlobSize, которая возвращает размер данного BLOB или MEMO поля.  

    function GetBlobSize(Field: TBlobField): Longint; 

    begin 

      with TBlobStream.Create(Field, bmRead) do 

      try 

        Result := Seek(0, 2); 

      finally 

        Free; 

      end

    end

    procedure TForm1.Button1Click(Sender: TObject); 

    begin 

    { This sets the Edit1 edit box to display the size of } 

    { a memo field named Notes.                           } 

      Edit1.Text := IntToStr(GetBlobSize(Notes)); 

    end


    22. Как осуществить поиск по неиндексированному полю в таблице?

    Вы можете добавить следующую функцию в ваш модуль и вызвать, например:

    Locate(Table1, Table1LName, 'Beman');

    Table1 — компонент TTable, Table1LName — TField, который вы добавили с помощью Fields Editor и 'Beman' - имя, которое вы хотите найти.


    23. Как узнать, что изменилась текущая запись?

    Событие TDataSource.OnDataChange когда State=dsBrowse.


    24. Как считать даты для вычисляемых полей?

    При вычислении дат важно удостовериться в том, что все используемые значения подходят по типу. В документации не совсем явно отражен тот факт, что на самом деле тип TDataTime эквивалентен типу Double, который можно использовать далее.

    В примере, D1 и D2 (поля в Table1) могут быть типа или Date, или TDateTime, а D3 — поле типа Integer.  

    procedure TForm1.Table1CalcFields(DataSet: TDataset); 

    var 

      T1, T2: TDateTime; 

    begin 

      Table1D1.AsDateTime := Date+2; 

    { or Table1D1.Value := Date+2; } 

      Table1D2.AsDateTime := Date-2; 

      T1 := Table1D1.AsDateTime; 

      T2 := Table1D2.AsDateTime; 

      Table1D3.AsInteger := Trunc(Double(T1) - Double(T2)); 

    end

    Компоненты и VCL

    1. Почему возникает ошибка компиляции при обращении к объекту Sender в обработчике события?

    Я в обработчике события OnChange для компонента TEdit пытаюсь получить содержимое его текстового буфера. Однако, следующая конструкция вызывает ошибку компиляции 'неизвестный идентификатор':

    Caption := Sender.Text;

    Если вы рассматривали декларацию, объект Sender имеет тип TObject, который является классом, который наследуется почти всеми остальными объектами. Вы, вероятно, пробуете обращаться к свойству, которое не определено в TObject, вроде Text или Caption. По этой причине, выражение Sender.Text вызовет ошибку, но если (для примера) вы знаете, что Sender имеет тип TEdit, тогда вы можете использовать выражение:

    Caption := (Sender as TEdit).Text;

    Если вы не уверены, что объект Sender будет всегда иметь данный тип, то рекомендуется предварительно проверить это:

    if Sender is TEdit then …


    2. Проблемы с полями класса типа TObject, TTable и т.д.

    Я объявляю поле класса как TTable, но при обращении к нему происходит ошибка.

    Дело в том, что в Delphi все экземпляры объектов, объявленых как class, являются динамическими. Соответственно поле MyTable, объявленное как


    type 

       TMyClass = class(TObject) 

       public 

         MyTable: TTable; 

         constructor Create; 

         destructor Destroy; override

       end;    

    является указателем на класс TTable, и должно быть инициализировано в конструкторе вашего объекта и соответственно разрушено в деструкторе следующим образом:

    constructor TMyClass.Create; 

    begin 

      MyTable := TTable.Create(nil); 

      MyTable.DatabaseName := 'DBDEMOS'

    end

    destructor TMyClass.Destroy; 

    begin 

      MyTable.Free; 

    end

    Подробнее см. Changes in Object Pascal Language в документации или on-line help.


    3. Как закрыть модальную форму (ShowModal)? И вообще, каков лучший способ закрыть любую форму?

    Вообще говоря, нужно вызывать метод Close для формы. Close вызывает событие OnClose (обработчик которого может решить, что форму нельзя закрывать, например, если имеются несохраненные данные). Close не освобождает память, связанную с формой, если вы, конечно, не поместите в обработчик события вызов метода Release.

    Если вы хотите уничтожить форму без вызова события OnClose, используйте метод Release. Этот метод работает подобно Free, но позволяет всем обработчикам событий данной формы закончить работу перед тем, как память будет освобождена.

    Модальные формы "прекращают свой модальный статус", когда вы устанавливаете свойство ModalResult формы в любое значение, отличное от нуля. Если вы поместите кнопку на модальную форму и установите свойство ModalResult для кнопки в некоторое значение, то, когда пользователь нажмет на эту кнопку, форма закроется с результатом, который вы определили. Этот результат можно узнать вызывая ShowModal как функцию. То есть:

    Result := Form.ShowModal;


    4. Перемещение существующих компонентов на TPanel, TGroup и т.п.

    Я поместил кнопку (или что-то другое) на форму, затем поместил панель, и решил переместить кнопку на панель, но ничего не получилось.

    Действительно, чтобы поместить кнопку на панель, необходимо на форму сначала поместить панель, выбрать ее, а затем уже помещать кнопку.

    Но и в вашей ситуации есть решение. Скопируйте (Copy) или вырежьте (Cut) нужный компонент, выберите панель, и сделайте вставку (Paste). Рекомендуется предварительно "подогнать" копируемый компонент в левый верхний угол формы, иначе компонент на панели может выпасть из "пределов видимости" панели (или любого другого группового компонента).

    Если компонент все-таки "выпал" из пределов видимости - найдите этот компонент в Инспекторе Объектов, и установите нужные значения его свойств Left и Top.

    Используя группы компонент можно огранизовать форму-шаблон, на которой можно складывать (например в Notebook) компоненты с предварительно заданными свойствами, отличными от стандартных. Это решение проще чем добавлять такие компоненты в палитру компонент — не увеличивается размер библиотеки компонентов DCL (Delphi 3.0 не считается), не загромождается палитра компонент.

    Учтите, что при таком копировании компонент их имена меняются на новые (Button1, Button2 и т.д.).


    5. Как можно добавить новый компонент на страницу TTabbedNoteBook во время выполнения программы? Как нужно определить свойство Parent для этого компонента?

    Для того, чтобы добавить компонент на страницу TabbedNotebook, свойству Parent нового компонента нужно присвоить указатель на требуемую страницу. Способ для доступа к любой странице TTabbedNotebook во время выполнения — массив свойств Objects у свойства Pages компонента TTabbedNotebook. Другими словами, страницы сохранены в виде объектов в свойстве Pages (тип TStringList). Пример демонстрирует создание кнопки TButton на второй странице TabbedNotebook1:  

    var 

      NewButton: TButton; 

    begin 

      NewButton := TButton.Create(Self); 

      NewButton.Parent := TWinControl(TabbedNotebook1.Pages.Objects[0]); 

    То же самое справедливо и для компонента TNotebook.


    6. Как включить символ & в надпись (Caption)?

    Попробуйте: &&


    7. Как сделать окно (TForm) без заголовка (Caption)?

    Попробуйте использовать следующий код:  

    constructor TPanelForm.Create(AOwner: TComponent); 

    { Вызов SetWindowLong API для изменения стиля окна главной формы. } 

    { Берется существующий стиль и убирается флаг WS_CAPTION          } 

    var  

      LStyle: Longint; 

    begin 

      inherited Create(AOwner); 

      BorderIcons := []; 

      LStyle := GetWindowLong(Handle, GWL_STYLE); 

      LStyle := LStyle and not WS_CAPTION; 

      SetWindowLong(Handle, GWL_STYLE, LStyle); 

    { Перерисуем окно } 

      ForceRepaint; 

    end

    procedure TPanelForm.ForceRepaint; 

    var 

      RectWnd: TRect; 

      WWidth, WHeight: Integer; 

    begin 

    { получаем размер окна } 

      GetWindowRect(Handle, RectWnd); 

      WWidth := RectWnd.Right-RectWnd.Left; 

      WHeight := RectWnd.Bottom-RectWnd.Top; 

    { Форсируем полную перерисовку. Это должен делать InvalidateRect() } 

    { но почему-то не делает  : (  } 

    { "сожмем" окно } 

      SetWindowPos(Handle, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE); 

    { восстановим окно } 

      SetWindowPos(Handle, HWND_TOP, 0, 0, WWidth, WHeight, SWP_NOMOVE); 

    end;  
     

    Можно поступить другим способом — выставить у формы свойство BorderStyle = bsNone, и написать следующий обработчик OnPaint:

    procedure TForm1.FormPaint(Sender: TObject); 

    begin 

      Canvas.Brush.Style := bsClear; 

      Canvas.Rectangle(0, 0, Width, Height); 

    end;    


    8. Почему некоторые компоненты типа TPanel и TEdit не имеют свойства Canvas?

    Все наследники TCustomControl имеют Canvas, однако, в большинстве случаев это свойство объявлено protected для предотвращения рисования 'чужаками' на компоненте. Наследники компонента всегда могут получить доступ к унаследованным protected свойствам (типа Canvas), но пользователь компонента — никогда.  

    type 

      TCanvasPanel = class(TPanel) 

      public 

        property Canvas; 

      end

    Если вы хотите рисовать на компоненте, у которого нет public свойства Canvas, то используйте, например, компонент TPaintBox: положите его на панель TPanel, сделайте Align = Client и рисуйте на TPaintBox.Canvas.


    9. Почему при уничтожении компонента в методе OnClick происходит ошибка?

    Допустим, вы поместили на форму кнопку, и создали метод OnClick в котором вызываете Button1.Free. Вы видите, что это метод формы — казалось бы, какие препятствия для правильного уничтожения кнопки?

    На самом деле Button1.OnClick является свойством и после запуска вашего приложения содержит адрес метода Form1.Button1Click. Именно кнопка вызывает этот метод как свой собственный. А это означает, что кнопка не может удалить себя в своем-же методе. Даже если вы попытаетесь удалить ссылку в OnClick:

    Button1.OnClick := nil;

    Button1.Free;

    то это не поможет — стек настроен на возврат в обработчик TButton, который и вызвал OnClick. Поскольку к моменту возврата объект разрушен — возникает GPF или Access Violation.


    10. Есть ли у TDBGrid события OnMouseDown, OnMouseUp и OnMouseMove?

    Они есть, но не объявлены published. Вы можете создать наследника TDBGrid и сделать их published.


    11. Поиск компонента в форме по имени.

    Я хочу делать текущими в форме произвольные компоненты. Как выставить фокус у конкретного компонента ясно - ListBox1.SetFocus. А если я хочу обращаться к некоему компоненту по имени (свойство Name)?

    Свойство TForm.Components — массив компонентов формы, который и нужен вам. Вы можете перемещаться по этому массиву пока не найдете компонент с нужным Name. Например:  

    procedure TForm1.DooDah; 

    var 

      Count: Integer; 

    begin 

      Count := 0; 

      while (Count < ComponentCount) and (Components[Count] <> 'Button1') do Inc(Count); 

      TButton(Components[Count]).SetFocus; 

    end;

    или еще проще:

    procedure TForm1.DooDah; 

    var 

      Target: TComponent; 

    begin 

      Target := FindComponent('Button1'); 

      TButton(Target).SetFocus; 

    end;    

    Оба этих примера показывают как найти компонент TButton с именем Button1, и вызвать его метод SetFocus.


    12. Как получить горизонтальный ScrollBar на ListBox?

    Пошлите сообщение LB_SETHORIZONTALEXTENT в ListBox. Например, сообщение может быть отослано в момент создания формы:  

    procedure TForm1.FormCreate(Sender: TObject); 

    begin 

      SendMessage(Listbox1.Handle, LB_SETHORIZONTALEXTENT, 1000, Longint(0)); 

    end


    13. Как определить текущую колонку и строку каретки в компоненте TMemo?

    Вы можете использовать сообщения Windows API EM_LINEFROMCHAR и EM_LINEINDEX для определения положения.  

    var 

      LineNum: Longint; 

      CharsBeforeLine: Longint; 

    begin 

      LineNum := SendMessage(Memo1.Handle, EM_LINEFROMCHAR, Memo1.SelStart, 0); 

      CharsBeforeLine := SendMessage(Memo1.Handle, EM_LINEINDEX, LineNum, 0); 

      Label1.Caption := 'Line ' + IntToStr(LineNum + 1) 

      Lebel2.Caption := 'Position ' + IntToStr(Memo1.SelStart - CharsBeforeLine + 1); 

    end;


    14. Постранична прокрутка TMemo, реализация Undo и определение строки курсора.

    Как прокрутить содержимое компонента TMemo?

    Приведенная ниже процедура предполагает, что фокус находится на Edit1 и осуществляет прокрутку в соответствии с нажатыми клавишами.  

    procedure TForm1.Edit1KeyDown(Sender: TObject; var Key: Word; Shift: TShiftState); 

    begin 

      if Key = VK_F8 then 

        SendMessage(Memo1.Handle,  { HWND для Memo } 

                    WM_VSCROLL,    { сообщение Windows } 

                    SB_PAGEDOWN,   { на страницу вниз } 

                    0)             { не используется } 

      else if Key = VK_F7 then SendMessage(Memo1.Handle, WM_VSCROLL, SB_PAGEUP, 0); 

    end;

    Если определено всплывающее (popup) меню для TMemo,и заданы клавиши для операций Cut, Copy, Paste, то я могу обрабатывать эти события, вызывая методы CutToClipboard, CopyToClipboard, и т.д. Однако, если я поместили пункт Undo в меню (обычно Ctrl+Z), то как дать знать TMemo, что нужно выполнить Undo?

    Если встроенного Undo достаточно, то это очень просто:

    Memo1.Perform(EM_UNDO, 0, 0);

    Для переключения свойства Enabled пункта меню Undo1:

    Undo1.Enabled := Memo1.Perform(EM_CANUNDO, 0, 0) <> 0;

    Как можно определить, на какой строке в TMemo находится курсор?

    Весь фокус в сообщении EM_LINEFROMCHAR. Попробуйте:  

    procedure TMyForm.BitBtn1Click(Sender: TObject); 

    var 

      ILine: Integer; 

    begin 

      ILine := Memo1.Perform(EM_LINEFROMCHAR, $FFFF, 0); 

    { Внимание: номера строк начинаются с нуля } 

      MessageDlg('Line Number: ' + IntToStr(ILine), mtInformation, [mbOK], 0); 

    end


    15. Как поместить BLOB Memo в компонент TMemo?

    Попробуйте так:  

    procedure TForm1.Button1Click(Sender: TObject); 

    var 

      S: TBlobStream; 

    begin 

      S := TBlobStream.Create(Table1BBBMemo, bmRead); 

      Memo1.Lines.LoadFromStream(S); 

      S.Free; 

    end

    где:

    1. Table1BBBMemo — имя поля BLOB Memo (TMemoField).

    2. Memo1 — имя компонента TMemo. Естественно, что этим же способом можно обмениваться информацией с BLOB-полями произвольного типа.


    16. Как показать содержимое Memo поля в TDBGrid?

    Используйте следующий код для обработки события OnDrawDataCell у TDBGrid. (Перед запуском программы создайте объект TMemoField для memo поля в Fields Editor).  

    procedure TForm1.DBGrid1DrawDataCell(Sender: TObject; const Rect: TRect; Field: TField; State: TGridDrawState); 

    var 

      P: array [0..1023] of Char; { MemoField buffer } 

      BS: TBlobStream; 

      S: string

    begin 

      if Field is TMemoField then 

      with (Sender as TDBGrid).Canvas do 

      begin 

    { Table1Notes is the TMemoField } 

        BS := TBlobStream.Create(Table1Notes, bmRead); 

        FillChar(P, SizeOf(P), #0); 

        BS.Read(P, SizeOf(P)); 

        BS.Free; 

        S := StrPas(P); 

    { remove carriage returns & line feeds } 

        while Pos(#13, S) > 0 do S[Pos(#13, S)] := ' '

        while Pos(#10, S) > 0 do S[Pos(#10, S)] := ' '

    { clear the cell } 

        FillRect(Rect); 

    { fill cell with memo data } 

        TextOut(Rect.Left, Rect.Top, S); 

      end

    end;    


    17. Не возникает событие TSpeedButton.OnDblClick.

    Я создаю событие на SpeedButton1.OnDblClick, но оно, похоже, вообще никогда не возникает. OnClick работает. Что делать?

    На самом деле работает, только в определенных ситуациях. Если вы помещаете на панель несколько кнопок, то по умолчанию они независимы и соответственно не фиксируются в нажатом состоянии. Поскольку одиночное нажатие мыши на кнопку отрабатывается немедленно, двойной щелчок мыши воспринимается как два нажатия и отпускания. Поэтому OnDblClick и не срабатывает.

    Если  же кнопки связаны в группу (GroupIndex <> 0), то они могут фиксироваться, и соответственно могут воспринимать двойной щелчок мыши.


    18. Как разделить обработку OnClick и OnDblClick? Ведь OnClick будет вызываться всегда, и перед DblClick.

    Именно так и происходит в Windows — посылаются оба сообщения. Для того чтобы обработать только какое-то одно событие необходимо чуть "задержать" выполнение OnClick. Сделать это можно следующим способом:  

    procedure TForm1.ListBox1Click(Sender: TObject); 

    var 

      Msg: TMsg; 

      TargetTime: Longint; 

    begin 

    { get the maximum time to wait for a double-click message } 

      TargetTime := GetTickCount + GetDoubleClickTime; 

    { cycle until DblClick received or wait time run out } 

      while GetTickCount < TargetTime do 

      if PeekMessage(Msg, ListBox1.Handle, WM_LBUTTONDBLCLK, WM_LBUTTONDBLCLK, WM_NOREMOVE) 

      then Exit; { Double click } 

      MessageDlg('Single clicked', mtInformation, [mbOK], 0); 

    end;    


    19. Как определить из обработчика события OnClick в Popup.MenuItem, для какого объекта это произошло?

    Используйте свойство PopupComponent компонента TPopupMenu для определения, где была нажата правая кнопка.  

    procedure TForm1.PopupItem1Click(Sender: TObject); 

    begin 

      Label1.Caption := PopupMenu1.PopupComponent.Name; 

    end

    Свойство ActiveControl для формы тоже можно использовать, однако, ActiveControl не обязательно является тем элементом, для которого произошло событие.


    20. Как использовать case, чтобы определить, какой объект вызвал процедуру?

    Используйте свойство Tag. Установите значение Tag свое у каждого объекта для опознания. (Использование констант, которые описывают объект — идеально подходит).  

    case (Sender as TComponent).Tag of 

      Button1ID: SomeProcedure; 

      Button2ID: AnotherProcedure; 

    end

    Таким образом вы можете обрабатывать события как от однотипных компонент, так и от компонент разного типа.


    21. Как обрабатывать события от множества однотипных компонентов.

    На моей форме находится примерно 10 кнопок. Я хочу обрабатывать нажатие на любую из них одним событием, но как их отличить внутри обработчика события?

    Для этого базовый класс VCL TComponent имеет поле Tag типа Longint. В момент разработки вы можете присвоить этому полю любое значение, а в момент исполнения использовать его (или переопределять). В вашей ситуации достаточно присвоить полю ButtonX.Tag значение от 1 до 10 (или от 0 до 9, как удобнее), а в обработчике написать примерно следующее:  

    procedure MyForm.Button1Click(Sender: TObject); 

    begin 

      case (Sender as TComponent).Tag of 

        1: {...}

        2: {...}

        3: {...}

      end

    end;    


    22. Использование TPanel в качестве "индикатора".

    Я пытаюсь использовать TPanel как индикатор процесса обновления БД.  Однако надпись на панели не обновляется пока не закончится цикл обработки БД. В цикле вызывается Panel.Caption := ...

    После присвоения Panel.Caption вызывайте Panel.Refresh или Application.ProcessMessages (второй вариант предпочтительней, так как позволяет перерисовать себя всем клмплнентам, которые в этом нуждаются).


    23. Включение и выключение подсказок (Hints) для всех элементов на форме.

    Если ваша форма содержит панель подсказки в нижней части формы, то вы можете определить подменю для этой панели, и выставлять Form.ShowHint в True или False в зависимости от состояния Checked элемента меню.

    Например, в TMenuItem.OnClick напишите:

    ShowHint := not (Sender as TMenuItem).Checked;

    В результате на локальном меню панели будет видно, включены подсказки для всех элементов или нет.


    24. Как в меню поместить bitmap?

    Можно поступить таким образом:  

    var 

      Bmp1: TBitmap; 

    begin 

      Bmp1 := TBitmap.Create; 

      Bmp1.LoadFromFile('C:\WHERE\B1.BMP'); 

      SetMenuItemBitmaps(MenuItemTest.Handle, 0, MF_BYPOSITION, Bmp1.Handle, Bmp1.Handle); 

    end

    Параметры:

    • MenuItemTest — имя пункта меню (горизонтальная строка)

    • 0,1 ... — позиция пункта меню, в который надо вставить BMP

    • первый Handle — для показа невыбранного пункта меню (Unchecked)

    • второй Handle — для выбранного (Checked). Они могут быть разные

    Код можно вставить в обработчик OnCreate для формы. При уничтожении меню TBitmap не уничтожается, это надо делать отдельно.


    25. Каким образом можно поместить двумерный массив в TImage?

    Представим, что данные находятся в массиве:

    TestArray: array [0..127, 0..127] of Byte;

    Картинка будет иметь размер 128×128 точек:

    Image1.Picture.Bitmap.Width := 128;

    Image1.Picture.Bitmap.Height := 128;

    Вызываем функцию Windows API для формирования bitmap:

    SetBitmapBits(Image1.Picture.Bitmap.Handle, SizeOf(TestArray), @TestArray);

    Image1.Refresh;  { для того, чтобы изменения отобразились }

    Однако, если вы используете свою палитру, то ее нужно создавать специально.


    26. Как из программы 'открыть' TComboBox?

    У TComboBox есть run-time свойство, не упомянутое в on-line help — DroppedDown.

    Для открытия ComboBox напишите:

    ComboBox1.DroppedDown := True;

    Естественно, False закроет его.


    27. Как заменить надпись 'Read only' в компонентах TSaveDialog и TOpenDialog?

    Попробуйте посмотреть в Windows API Help разделы, связанные с lpTemplateName. Вообще говоря, вы можете заменить стандартный Open Dialog Box своим собственным шаблоном.


    28. Проблема в использовании компонента TCustomGrid.

    Делаю так:

    1. Создаю новый компонент при помощи Эксперта Компонент

    2. Имя класса TSampleCalendar

    3. Имя родителя TCustomGrid

    4. Использую страницу 'Samples'

    5. Сохраняю модуль с именем CALSAMP.PAS

    6. Подключаю к Палитре компонент

    7. Создаю форму, помещаю новый компонент на форму и получаю Runtime Error 210 В чем дело?

    Проблема в том, что TCustomGrid имеет метод DrawCell, который является абстрактным. То, что его безусловно надо переписывать у любого наследника TCustomGrid, к сожалению, не отражено в документации. Создайте этот метод (пусть даже пустой) и ваша проблема исчезнет.


    29. Как установить формат для поля таблицы?

    В Fields Editor выберите поле для форматирования. Используя свойства DisplayFormat и EditFormat сделайте то, что нужно. DisplayFormat работает для поля, на которое не установлен фокус. EditFormat работает для поля, на которое фокус установлен. Форматирование аналогично первому параметру в функции FormatFloat, но без скобок.


    30. Можно ли использовать клавишу ENTER при вводе данных для перехода от поля к полю?

    Используйте данный код для события OnKeyPress компонента TEdit.  

    procedure TForm1.Edit1KeyPress(Sender: TObject; var Key: Char); 

    begin 

      if Key = #13 then 

      begin 

        SelectNext(Sender as TWinControl, True, True); 

        Key := #0

      end

    end

    Теперь Enter ведет себя как Tab. Затем, выберите все объекты, которые должны вести себя как Edit1 (за исключением кнопок) и в Object Inspector установите обработчик OnKeyPress в Edit1KeyPress. Каждый выбранный вами объект воспринимает Enter как Tab. Если вы хотите обрабатывать событие на уровне формы (а не в каждом отдельном компоненте), уберите обработчики события у всех компонент и создайте FormKeyPress — обработчик OnKeyPress для формы:  

    procedure TForm1.FormKeyPress(Sender: TObject; var Key: Char); 

    begin 

      if Key = #13 then 

      begin 

        SelectNext(Sender as TWinControl, True, True); 

        Key := #0

      end

    end

    Все объекты на форме будут воспринимать Enter как Tab.

    Общие вопросы по Delphi и данному FAQ (часть 3)

    Object Pascal и Windows API

    1. Как работает информация времени выполнения (RTTI)?

    Имеются два новых оператора: as и is. as  — оператор защищенного преобразования типов (typecasting). Вы можете использовать его, чтобы заставить компилятор преобразовать объект из одного типа в другой, но, если в во время выполнения эти типы окажутся несовместимыми, то вы получите ошибку. Hапример, если вы имеете класс TSport, с потомоками TBasketball и TFootball, вам может потребоваться переменная типа TSport; далее может так случиться, что в программе эта переменная будет фактически содержать экземпляр типа TFootball. Тогда вы можете обратиться к этой переменной

    (MySport as TFootball)

    чтобы получить доступ к специфическим свойствам из типа TFootball. Однако, если вы ошиблись и на самом деле это экземпляр типа TBasketball, то при обращении к несуществующим свойствам будет возникать ошибка. Оператор is определяет, принадлежит ли экземпляр объекта к данному классу, либо к классу одного из его предков, и используется для проверки, сработает ли преобразование типов с данным объектом. Если вы имеете переменную MySport типа TSport, и в настоящее время она содержит экземпляр TBasketball, тогда следующие выражения истинны:

    (MySport is TSport)

    (MySport is TBasketball)

    not (MySport is TFootball)

    Следует иметь ввиду, что компилятор разрешает использовать данные конструкции только для выполнения преобразования типов, связанных родственными отношениями. Так, конструкция (Button1 as TEdit) (переменная Button1 имеет тип TButton) вызовет ошибку компиляции, так как ни при каких условиях не может быть выполнено преобразование типов от TButton к TEdit или наоборот. Комбинация двух операторов может привести к выражению типа следующего :

    function PlayerGoodness(var MySport: TSport): Integer; 

    begin 

      if (MySport is TBasketball) then 

        Result := (MySport as TBasketball).ReboundShots 

      else if (MySport is TFootball) then 

        Result := (MySport as TFootball).TotalYardage; 

    end;

    Также, базовый класс TObject имеет набор методов, которые возвращают информацию, созданную компилятором в момент компиляции текста для поддержки RTTI. Hапример, метод TObject.ClassName возвращает имя класса любого объекта, наследованного от TObject. Hапример, TButton.ClassName вернет значение 'TButton'


    2. Как работает обработка исключительных ситуаций в Delphi?

    Основная структура выглядит примерно так:

    P := New(BigThing); 

    try 

      try 

        Proc1(P); 

        Proc2(P); 

      except 

        Handle(P); 

        raise

      end

    finally 

      Dispose(P); 

    end;

    Первая строка распределяет большой блок памяти. Затем, в блоке try, выполняется несколько операторов, каждый из которых может вызвать ошибку, или, другими словами, "вызвать исключительную ситуацию". Если возникает ошибка, оставшаяся часть блока try пропускается, и выполняются блоки except и finally. Если ошибок нет, то после выполнения всех операторов в блоке try выполнится блок finally. В любом случае, блок памяти будет освобожден. Блок try … finally ловит все, включая Windows GPF или Access Violation. Обратите внимание на вызов raise в блоке try … except. Он снова вызывает исключительную ситуацию, которая вызовет сообщение об ошибке после того, когда закончится блок finally. Если не вызвать raise, то считается, что вы обработали исключительную ситуацию самостоятельно в пределах блока except.


    3. Есть ли простой способ перехватить exception?

    Создайте метод для формы, перехватывающий исключения. Этот метод будет вызываться обработчиком OnException объекта Application. В вашем методе проверьте, тот ли это исключение, что вы ожидаете, например EDatabaseError. Почитайте on-line help для события OnException. Там есть информация, как вызвать собственный метод для события.  

    procedure TForm1.MyExcept(Sender: TObject; E: Exception); 

    begin 

      if E is EDatabaseError then MessageDlg('Поймали exception', mtInformation, [mbOk], 0) 

    { это не то, сделать raise } 

      else raise E; 

    end

    procedure TForm1.FormCreate(Sender: TObject); 

    begin 

      Application.OnException := MyExcept; 

    { здесь вы указываете, что событие OnException выполнит ваш метод } 

    end


    4. Delphi используют строки в стиле Pascal или C?

    И те и другие. Delphi имеет два различных набора функций манипулирования строками, один - для PChar; но в Delphi также есть функция MessageDlg, которая принимает строки типа Pascal.

    Delphi 2.0 добавляет так называемые длинные строки (AnsiString), которыми можно манипулировать как обычными строками в Pascal, но они имеют динамически изменяющийся размер и могут быть размером до 4Гбайт. Можно выполнять преобразования от PChar к AnsiString и наоборот. Старый строковый тип теперь называется ShortString. По умолчанию кличевое слово string соответствует типу AnsiString.


    5. Есть ли в Delphi битовые множества?

    В явном виде битовых множеств в языке Object Pascal нет. Но вместо этого можно использовать обычные множества, которые на самом деле и хранятся как битовые. Если множество вам нужно для проверки, установлен ли какой то бит в слове (байте и т.д.) можно попробовать такую конструкцию:

    type 

      PByteSet = ^TByteSet; 

      TByteSet = set of Byte; 

    var 

      W: Word; 

    ... 

    { если бит 3 в слове W установлен, тогда ... } 

      if 3 in PByteSet(@W)^ then ... 

    ... 

    В Delphi 2.0 есть специальный класс TBitSet, который ведет себя как битовое множество.Для Delphi 1.0 вы можете написать такой класс самостоятельно.


    6. Проблема с числом типа Single в DLL.

    Я написал на C++ DLL, в которой у меня функция использует число типа float, передал из Delphi число типа Single и получил GPF 'Invalid Opcode'. Что неправильно?

    Если вы используете числа с плавающей точкой, лучше передавать их не по значению, а по ссылке (указатель в C++). Вероятно DLL написана на MS Visual C++, так как Microsoft и Borland используют разные соглашения о передаче параметров при работе с сопроцессором. В случае Borland C++ и Delphi должны использовать одинаковый способ передачи параметров и значений (через стек сопроцессора). В любом случае вместо Single лучше использовать Double (double или long float в C++), так как вообще говоря, реальный тип, который соответствует типу Single точно не определен и может измениться в будущем.

    7. Как заставить приложение Delphi отвечать на сообщения Windows?

    Используем сообщение WM_WININICHANGED в качестве примера. Объявление метода в TForm позволит вам обрабатывать сообщение WM_WININICHANGED:

    procedure WMWinIniChange(var Message: TMessage); message WM_WININICHANGE;

    Код в implementation может выглядеть так:

    procedure TForm1.WMWinIniChange(var Message: TMessage); 

    begin 

      inherited

    { ... ваша реакция на событие ... } 

    end;

    Вызов inherited метода очень важен. Обратите внимание также на то, что для функций, объявленных с директивой message (обработчиков событий Windows) после inherited нет имени наследуемой процедуры, потому что она может быть неизвестна или вообще отсутствовать (в этом случае вы в действительности вызываете процедуру DefaultHandler).


    8. Как обработать события от других приложений?

    Попробуйте сделать это следующим образом:

    type 

      TForm1 = class(TForm) 

      ... 

      private 

        procedure WMNCActivate(var Msg: TMessage); message WM_NCACTIVATE; 

      end

    procedure TForm1.WMNCActivate(var Msg: TMessage); 

    begin 

    { здесь обработка принятых событий } 

    end


    9. Как перехватить сообщения Windows и обработать их перед тем, как выполнится строка Application.Run?

    Пример проекта показывает, как получить сообщения Windows в данном случае. Это редкий случай, в большинстве случаев переопределение процедуры Application.OnMessage будет делать то же самое.

    program Project1; 

    uses 

      Forms, 

      Unit1 in 'UNIT1.PAS' { Form1 }

      Messages, WinTypes, WinProcs, 

    {$R *.RES} 

    var 

      OldWndProc: TFarProc; 

    function NewWndProc(hWndAppl: HWnd; Msg, wParam: Word; lParam: Longint): Longint; export

    begin 

    { default WndProc return value } 

      Result := 0; 

    { handle messages here; the message number is in Msg } 

      Result := CallWindowProc(OldWndProc, hWndAppl, Msg, wParam, lParam); 

    end

    begin 

      Application.CreateForm(TForm1, Form1); 

      OldWndProc := TFarProc(GetWindowLong(Application.Handle, GWL_WNDPROC)); 

      SetWindowLong(Application.Handle, GWL_WNDPROC, Longint(@NewWndProc)); 

      Application.Run; 

    end


    10. Проблема с DragDrop для внешних программ.

    Я пишу небольшую программку — "мусорную корзину". В FormCreate вызывается DragAcceptFiles(HANDLE, True). Проблема в том, что когда размер окна восстанавливается и затем минимизируется Drag and Drop перестает работать. Я безуспешно пробовал помещать DragAcceptFiles в разные методы формы. Однако если сделать вызов DragAcceptFiles(Application.Handle, True) в MainForm.Create, то все работает. Как перехватить событие WM_DROPFILES?

    Это можно сделать так:

    type 

      TMainForm = class(TForm) 

      ... 

        procedure FormCreate(Sender: TObject); 

      private 

        procedure DropFiles(var Msg : TWMDropFiles); message WM_DROPFILES; 

      end


    procedure TMainForm.DropFiles(var Msg : TWMDropFiles); 

    begin 

      DragQueryPoint(Msg.Drop, Point); 

      NrOfFiles := DragQueryFile(Msg.Drop, Word(-1), FileName, BufSize); 

      DragQueryFile(Msg.Drop, 0, FileName, BufSize); 

    end

    procedure TMainForm.FormCreate(Sender: TObject); 

    begin 

      DragAcceptFiles(Handle, True); 

    end

    Подробнее о перехвате событий Windows см. Главу 7 руководства Component Writers Guide.


    11. Как обрабатывать WM_DROPFILES (Drag/Drop)?

    Следующий код показывает как обрабатывать это событие. Обрабатываются имена всех "брошенных" файлов. Для загрузки каждого файла вызывается CreateChild(FName). В обработчике OnCreate данной формы вы должны вызвать DragAcceptFiles.


    type 

      TFrameForm = class(TForm) 

      ... 

      protected 

        procedure WMDropFiles(var Msg: TMessage); message WM_DROPFILES; 

      end

    procedure TFrameForm.WMDropFiles(var Msg : TMessage); 

    var 

      I, N, Size: Word; 

      FName: string

      HDrop: Word; 

    begin 

      HDrop := Msg.WParam; 

      N := DragQueryFile(HDrop, $FFFF, nil, 0); 

      for I := 0 to (N-1) do 

      begin 

        Size := DragQueryFile(HDrop, I, nil, 0); 

        if Size < 255 then { 255 char. string limit - not really a problem } 

        begin 

          FName[0] := Chr(Size); 

          DragQueryFile(HDrop, I, @FName[1], Size+1); 

          CreateChild(FName); 

        end

      end

      Msg.Result := 0; 

      inherited

    end


    12. Как может выделить время CPU другим задачам , подобно "DoEvents" в VB?

    Эквивалент в Delphi — Application.ProcessMessages.

    Если вы выполняете долгие вычисления, то вызов данного метода позволит в Win 16 выполняться параллельно другим приложениям, а в Win 32 - корректно перерисовываться вашему приложению.


    13. В каком порядке происходят события при создании и показе окна?

    При создании окна обработчики событий выполняются в следующем порядке:

    • OnCreate

    • OnShow

    • OnPaint

    • OnActivate

    • OnResize

    • OnPaint (снова)


    14. UpCase для русского языка.

    Данная функция (UpCase) производит преобразование только латинских символов в верхний регистр. Для правильного преобразования необходимо использовать функции Windows API, поскольку именно Windows должна "знать" о кодировке национальных символов. Причем к конфигурации BDE кодровка Windows не имеет никакого отношения — имея английские Windows без русификатора и выставив в BDE кодировку Paradox ANSII Cyrillic нормальных русских букв получить не удастся.

    А функции для преобразования следующие — OemToAnsi, AnsiToOem, OemToAnsiBuf, AnsiToOemBuf в Win16 (модуль WinProcs) и OemToChar, CharToOem, OemToCharBuf и CharToOemBuf в Win32 (модуль Windows)..


    15. Приложение, написанное на Delphi, не запускается минимизированным.

    Проверьте глобальную переменную CmdShow для того чтобы определить, в каком состоянии запускается приложение, и модифицируйте ее как вам необходимо:

    procedure TForm1.FormCreate(Sender: TObject); 

    begin 

      if CmdShow = SW_SHOWMINNOACTIVE then WindowState := wsMinimized; 

    end;

    Например, если необходимо запускать приложение либо минимизированным, либо максимизированным, используйте следующий код: 

    procedure TForm1.FormCreate(Sender: TObject);

    begin 

      if CmdShow = SW_SHOWMINNOACTIVE then WindowState := wsMinimized 

      else WindowState := wsMaximized; 

    end;


    16. Объясните разницу в помещении uses в секцию interface или implementation.

    Секция interface — интерфейсная. Туда попадают объявления констант, типов (в т.ч. и объектов или классов) переменных, процедур и функций. Поэтому для этой части uses должен содержать ссылки на те модули, которые используются для объявлений в этой части.

    Секция implementation — описание реализации интерфейсной части, здесь в uses должны быть упомянуты те модули, которыми вы пользуетесь для написания кода. Например, Вы хотите в модуле пользоваться функциями API Windows, для этого добавьте в объявлении implementation строку uses WinTypes, WinProcs; или uses Windows;. Таким образом, вы явно указываете что данными модулями будете пользоваться только в секции реализации.

    Конечно, можно упоминать модули только в части interface, но правильная расстановка имен модулей в соответствующем uses гарантирует исключение циклических ссылок, а также улучшает читаемость программы.


    17. Как спрятать окна MDI Child?

    Я пытаюсь это сделать, выставляя Form1.Visible := False, но это не помогает.

    Windows не позволяет прятать окна MDI Child.


    18. Как убрать заголовок у формы MDIChild?

    Как убрать заголовок (Caption) из MDIChild?

    Для MDIChild установка свойства BorderStyle := bsNone не убирает заголовок. Это можно сделать так:

    procedure TMDIChildForm.CreateParams(var Params: TCreateParams); 

    begin 

      inherited CreateParams(Params); 

      Params.Style := Params.Style and (not WS_CAPTION); 

    end;


    19. Сохранение данных в Clipboard.

    Мне нужно использовать clipboard для сохранения данных в собственном формате и я хочу для этого написать набор процедур ввода/вывода с использованием потоков (streams). Возможно ли создать объект TMemoryStream, эаполнить его и поместить в Clipboard?

    Не только возможно, именно так поступают функции Clipboard.GetComponent и Clipboard.SetComponent. Сначала вы должны зарегистрировать свой собственный формат данных для Clipboard с помощью функции RegisterClipboardFormat:

    CF_MYFORMAT := RegisterClipboardFormat('My Format Description'); Далее вы должны выполнить шаги:

    1. Создать поток (memory stream) и записать туда данные.

    2. Создать глобальный буфер в памяти и скопировать поток туда.

    3. Вызвать Clipboard.SetAsHandle(), чтобы поместить буфер в Clipboard.

    Пример:

    var 

      hBuf: THandle; 

      Bufptr: Pointer; 

      MStream: TMemoryStream; 

    begin 

      MStream := TMemoryStream.Create; 

      try 

      { write your data to the stream } 

        hBuf := GlobalAlloc(GMEM_MOVEABLE, MStream.Size); 

        try 

          BufPtr := GlobalLock(hBuf); 

          try 

            Move(MStream.Memory^, BufPtr^, MStream.Size); 

            Clipboard.SetAsHandle(CF_MYFORMAT, hBuf); 

          finally 

            GlobalUnlock(hBuf); 

          end

        except 

          GlobalFree(hBuf); 

          raise

        end

      finally 

        MStream.Free; 

      end

    end

    Внимание: не уничтожайте буфер, созданный с GlobalAlloc. Поскольку вы поместили его в Clipboard, это уже дело clipboard'а его уничтожить. Опять же, получая буфер из Clipboard, не уничтожайте этот буфер - просто сделайте копию содержимого.

    Для обратного получения потока и данных, сделайте что-нибудь вроде этого:

    var 

      hBuf: THandle; 

      BufPtr: Pointer; 

      MStream: TMemoryStream; 

    begin 

      hBuf := Clipboard.GetAsHandle(CF_MYFORMAT); 

      if hBuf <> 0 then 

      begin 

        BufPtr := GlobalLock(hBuf); 

        if BufPtr <> nil then 

        try 

          MStream := TMemoryStream.Create; 

          try 

            MStream.WriteBuffer(BufPtr^, GlobalSize(hBuf)); 

            MStream.Position := 0; 

          { read your data from the stream } 

          finally 

            MStream.Free; 

          end

        finally 

          GlobalUnlock(hBuf); 

        end

      end

    end


    20. Что означает Key<>#0 ? 

    В исходном тексте одного из компонентов третьих фирм я увидел строку:

    if Key <> #0 then inherited KeyPress(#0);

    В Windows виртуальные коды находятся в диапазоне 1-145 (Dec). Зачем нужна такая проверка?

    В соответствии с соглашением Windows код клавиши #0 означает отсутствие реального нажатия. Управление в данную точку программы могло попасть, например вследствие прямого вызова, а не нажатия клавиши или же нажатие уже было обработано предком, вследствие чего код нажатой клавиши был сброшен в 0.


    21. Аналог процедуры TP/BP Delay.

    procedure TForm1.Delay(MSecs: Longint); 

    var 

      FirstTick: Longint; 

    begin 

      FirstTick := GetTickCount; 

      repeat 

        Application.ProcessMessages; 

      until GetTickCount - FirstTick >= MSecs; 

    end

    В Win32 API существуют также функции Sleep и SleepEx.


    22. Каким образом создать форму, которую можно таскать за поле?

    Как сделать форму (окно), которое перетаскивается не за заголовок (Сaption), а за все поле ?

    Нужно обрабатывать сообщение WM_NCHITTEST:

    type 

      TForm1 = class(TForm) 

      ... 

      private 

        procedure WMNCHitTest(var M: TWMNCHitTest); message WM_NCHITTEST; 

      end

    procedure TForm1.WMNCHitTest(var M: TWMNCHitTest); 

    begin 

      inherited;                  { вызов унаследованного обработчика      } 

      if M.Result = htClient then { Мышь сидит на окне?                    } 

         M.Result := htCaption;   { Если да - то пусть Windows думает, что } 

                                  { мышь на caption bar                    } 

    end

    Примечание: окно можно сделать вообще без Сaption.


    23. Как программно спрятать или показать заголовок у формы?

    Как программно спрятать или показать заголовок (Caption) у формы?

    Вы можете попробовать следующее:

    procedure TForm1.HideTitlebar; 

    var 

      Save: Longint; 

    begin 

      if BorderStyle=bsNone then Exit; 

      Save := GetWindowLong(Handle, GWL_STYLE); 

      if (Save and WS_CAPTION) = WS_CAPTION then 

      begin 

        case BorderStyle of 

          bsSingle, bsSizeable: 

            SetWindowLong(Handle, GWL_STYLE, Save and (not WS_CAPTION) or WS_BORDER); 

         bsDialog: 

            SetWindowLong(Handle, GWL_STYLE, Save and (not WS_CAPTION) or DS_MODALFRAME or WS_DLGFRAME); 

        end

        Height := Height-GetSystemMetrics(SM_CYCAPTION); 

        Refresh; 

      end

    end

    procedure TForm1.ShowTitlebar; 

    var 

      Save: Longint; 

    begin 

      if BorderStyle = bsNone then Exit; 

      Save := GetWindowLong(Handle, GWL_STYLE); 

      if (Save and WS_CAPTION) <> WS_CAPTION then 

      begin 

        case BorderStyle of 

          bsSingle, bsSizeable: 

            SetWindowLong(Handle, GWL_STYLE, Save or WS_CAPTION or WS_BORDER); 

          bsDialog: 

            SetWindowLong(Handle, GWL_STYLE, Save or WS_CAPTION or DS_MODALFRAME or WS_DLGFRAME); 

        end

        Height := Height + GetSystemMetrics(SM_CYCAPTION); 

       Refresh; 

      end

    end

    24. Как сделать приложение модальным?

    Мне нужно сделать приложение модальным, для того чтобы обезопасить систему и в то же время позволить работать с программой.

    Ok, пара предложений на эту тему:

    1. Создайте форму, занимающую весь экран (maximized) без системных кнопок (Maximize, Minimize, System)

    2. В обработчике FormDeactivate для формы вызовите метод SetFocus — это предотвратит Ctrl+Esc:

    Form1.SetFocus;

    3. В обработчике события FormActivate, нужно присвоить метод Deactivate для приложения:

    Application.OnDeactivate := FormDeactivate;

    4. Создайте всплывающее меню TPopupMenu с единственным пунктом. В свойствах данного компонента нужно установить Visible=False. Создайте процедуру для этого пункта меню, и в теле поставьте две фигурные скобки {} (для того, чтобы Delphi не удалил эту процедуру)

    5. Присвойте созданное Popup-меню форме (св-во PopupMenu)

    6. Задайте горячую клавишу (shortcut) для Popup-меню в методе FormActivate как показано ниже:

    NullItem1.ShortCut := ShortCut(VK_Tab, [ssAlt]);

    (NullItem1 нужно заменить на название созданного вами объекта — пункта меню)

    Шаги 4-6 предотвращают переход на приложение по Alt-Tab.


    25. Как изменить шрифт у Application.Title (заголовка приложения)?

    Никак. Это ограничение Windows — вы не можете изменить шрифт ни у одного заголовка ни у приложения, ни у окна. Для окна можно предложить следующее — создать свое окно без заголовка (Caption) и рамки, которое будет само выводить нужную надпись нужным шрифтом и одновременно будет способно изменять свои размеры.


    26. Каким образом (желательно не специфичным для Delphi) узнать, открыто меню или нет?

    Вот так:

    type 

      TForm1 = class(TForm) 

        MainMenu1: TMainMenu; 

        Item01: TMenuItem; 

        Item11: TMenuItem; 

        Item21: TMenuItem; 

      private 

        { Private declarations } 

      public 

        procedure WMMenuSelect(var M: TWMMenuSelect); message WM_MENUSELECT; 

      end

    implementation 

    {$R *.RES} 

    procedure TForm1.WMMenuSelect(var M: TWMMenuSelect); 

    begin 

      inherited

    { Этот Beep сигнализирует вообще об открытии меню } 

      MessageBeep(MB_ICONASTERISK); 

    { А зтот Beep - только о выборе в меню нового Item } 

      if M.Menu = MainMenu1.Handle then MessageBeep(MB_ICONASTERISK); 

    end

    end.

    Разное

    1. Передача переменной в отчет ReportSmith.

    Следующий код показывает, как передать переменную в отчет.

    В примере строковой переменной отчета 'City' присваивается значение 'Bombey'. Подразумевается, что есть готовый отчет с данной переменной. Поместите компонент TReport на форму и установите требуемые свойства для вызова печати отчета. Напишите обработчик OnClick для кнопки Button1 на форме (кнопка — для простоты):

    procedure TForm1.Button1Click(Sender: TObject); 

    begin 

      Report1.InitialValues.Clear; 

      Report1.InitialValues.Add('@City=<Bombey>'); 

      Report1.Run; 

    end;


    2. Как получить русские буквы в DBD?

    Имя шрифта для отображения русских букв берется из файла PDOXWIN.INI секция [Properties] строка SystemFont. Если очень хочется, то можно исправить имя 'PDOXWIN.INI' на 'DBD.INI' в файле DBSRV.DLL (он лежит там же где и DBD.EXE) по смещению $E9D8 (не забудьте после 'DBD.INI' поставить шестнадцатеричный ноль), и в секции [Properties] файла DBD.INI добавить строку типа

    SystemFont = Courier New Cyr

    По умолчанию имя фонта для отображения русских букв — Arial.

    Действительно, если у Вас Pan Euro или русская версия Windows95, то DBD не будет показывать шрифты Cyr в Preferences/General/Default system font.

    Решить эту проблему можно двумя способами:

    1. записать в каталог WINDOWS/FONTS шрифты Arial Cyr от русских Windows и сделать ShutDown. После загрузки Arial Cyr будет доступен для выбора.

    2. поменять шрифт в Registry вручную например на MS Sans Serif - HKEY_CURRENT_USER/SOFTWARE/Borland/DBD/7.0/Preferences/Properties ключ SystemFont.


    3. Как печатать отчеты из приложения Delphi без использования ReportSmith?

    1. Лучше всего использовать специализированные генераторы отчетов в виде компонентов, например QuickReport или Ace Reporter.

    2. Можно использовать печать формы, например: Form1.Print.

    3. Можно использовать свойство Canvas объекта Printer.


    4. Как узнать количество точек на дюйм для принтера?

    VertPixelsPerInch := GetDeviceCaps(Printer.Handle, LogPixelsX);

    HorzPixelsPerInch := GetDeviceCaps(Printer.Handle, LogPixelsY);


    5. Как определить, приложение запущено из под Delphi IDE или как отдельный файл?

    Для этого следует проверить существование определенных окон:

    Delphi 1.0

    function DelphiLoaded: Boolean; 

      function WindowExists(ClassName, WindowName: string): Boolean; 

      var 

        PClassName, PWindowName: PChar; 

        AClassName, AWindowName: array [0..63] of Char; 

      begin 

        if ClassName = '' then PClassName := nil 

        else PClassName := StrPCopy(@AClassName[0], ClassName); 

        if WindowName = '' then PWindowName := nil 

        else PWindowName := StrPCopy(@AWindowName[0], WindowName); 

        Result :=  FindWindow(PClassName, PWindowName) <> 0; 

      end

    begin 

      Result := WindowExists('TPropertyInspector', 'Object Inspector'

            and WindowExists('TMenuBuilder', 'Menu Designer'

            and WindowExists('TApplication', 'Delphi'

            and WindowExists('TAlignPalette', 'Align'

            and WindowExists('TAppBuilder', ''); 

    end

    Delphi 2.0

    function DelphiLoaded: Boolean; 

      function WindowExists(ClassName, WindowName: string): Boolean; 

      begin 

        Result :=  FindWindow(PChar(ClassName), PChar(WindowName)) <> 0; 

      end

    begin 

      Result := WindowExists('TPropertyInspector', 'Object Inspector'

            and WindowExists('TMenuBuilder', 'Menu Designer'

            and WindowExists('TApplication', 'Delphi'

            and WindowExists('TAlignPalette', 'Align'

            and WindowExists('TAppBuilder', ''); 

    end

    Другой вариант для Delphi 1.0, работает только в EXE файлах (не в DLL).

    function InIDE: Boolean; 

    begin 

      Result := Bool(PrefixSeg) and Bool(PWordArray(MemL[DSeg:36])^[8])); 

    ;end   


    6. Что нужно предусмотреть при разработке приложения, которое будет работать при различном разрешении дисплея?

    На ранней стадии создания приложения решите для себя хотите ли вы позволить форме масштабироваться. Преимущество немасштабируемой формы в том, что ничего не меняется во время выполнения. В этом же заключается и недостаток (ваша форма может бать слишком маленькой или слишком большой в некоторых случаях).

    1. Если вы не собираетесь делать форму масштабируемой, установите свойство Scaled=False и дальше не читайте.

    2. В противном случае Scaled=True.

     1. Установите AutoScroll=False. AutoScroll = True означает 'не менять размер окна формы при выполнении' что не очень хорошо выглядит, когда содержимое формы размер меняет.

     2. Установите шрифты в форме на самые распространенные TrueType шрифты, например Arial, Times New Roman, Courier. Если вдруг выбранного шрифта не окажется на пользовательском компьютере, то Windows выберет альтернативный шрифт из того же семейства. Этот шрифт может не совпадать по размерус исходным, что вызовет проблемы.

     3. Установите св-во Position в любое значение, отличное от poDesigned. poDesigned оставляет форму там, где она была во время дизайна, и, например, при разрешении 1280×1024 форма может оказаться в левом верхнем углу и совершенно за экраном при 640×480.

     4. Оставляйте по крайней мере 4 точки между компонентами, чтобы при смене положения границы на одну позицию компоненты не "наезжали" друг на друга.

     5. Для однострочных меток TLabel с выравниванием alLeft или alRight установите AutoSize=True. Иначе AutoSize=False. Убедитесь, что достаточно пустого места у TLabel для изменения ширины фонта — 25% пустого места многовато, зато безопасно. При AutoSize=False Убедитесь, что ширина метки правильная, при AutoSize=True убедитесь, что есть свободное место для роста метки.

     7. Для многострочных меток (word-wrapped labels), оставьте хотя бы одну пустую строку снизу.

     8. Будьте осторожны при открытии проекта в среде Delphi при разных разрешениях. Свойство PixelsPerInch меняется при открытии формы. Лучше тестировать приложения при разных разрешениях, запуская готовый скомпилированный проект, а редактировать его при одном разрешении. Иначе это вызовет проблемы с размерами. Не изменяйте свойство PixelsPerInch самостоятельно!

     9. В общем, нет необходимости тестировать приложение для каждого разрешения в отдельности, но стоит проверить его на 640x480 с маленькими и большими шрифтами и на более высоком разрешении перед продажей.

     10. Уделите пристальное внимание принципиально однострочным компонентам типа TDBLookupCombo. Многострочные компоненты всегда показывают только целые строки, а TEdit покажет урезанную снизу строку. Каждый компонент лучше сделать на несколько точек больше. Даже при выполнении перечисленных инструкций, у вас могут возникнуть проблемы при переходе, например от Large fonts к Small fonts в Windows 95 при одном и том же разрешении. Бороться с этим помогают специально для этого разработанные компоненты. Если же вы решите самостоятельно изменять размеры компонентов, лежащих на форме, то вам могут помочь методы TCanvas.TextWidth и TCanvas.TextHeight.


    7. Конвертация ICO в BMP.

    Я создают toolbar, у меня есть иконки, но нет картинок в виде bitmap. Помогите!

    Для преобразования файлов из одного формата в другой лучше всего иметь что-нибудь вроде HiJaak, который может преобразовывать форматы напрямую. Однако, будем считать, что у вас нет ничего, кроме Windows и Delphi. Следующая процедура может использоваться чтобы преобразовывать иконку в формат Windows Bitmap:

    1. Покажите на экране иконку. Не имеет значения, как вы это сделаете.

    2. Нажмите Alt-PrintScreen, чтобы скопировать текущее окно в буфер Clipboard.

    3. Загрузите Paintbrush и сделайте Edit/Paste.

    4. Выберите нужный кусок изображения и сделайте Edit/Copy. Перейдите к пункту Options/Image Attributes и установите размер области 32x32 точки.

    5. Снова сделайте Edit/Paste.

    6. Сохраните результат как BMP файл.

    Лучше всего для редактирования и создания ресурсов (икон, картинок и т.п.) подходит Resource Workshop. Он включен в состав пакетов Borland Pascal 7.0 или Borland C++ 4.5, а также интегрирован в Borland C++ 5.0.

    В Delphi 1.0 есть специальный файл (X:\DELPHI\BIN\WORKOPT.DOS) который необходимо поместить в каталог, где находится Workshop — в этом случае последний будет "понимать" ресурсы, создаваемые Delphi 1.0 (например *.DCR).


    8. Когда используется свойство Glyph, как узнать, какой цвет прозрачный?

    Delphi всегда принимает, что цвет пикселя в левом нижнем углу картинки является фоновым цветом и должен отображаться на экране как прозрачный. Это нигде не документировано, но если у вас есть исходники VCL, вы можете посмотреть код в BUTTONS.PAS .


    9. Как отобразить bitmap в 256 цветах?

    Как подгрузить 256 цветный bitmap из ресурса и отобразить его в нормальной палитре?

    Обычно это делается следующим образом. Код Вадима Пузанова (Красноярск).

    Image1.Bitmap.Handle := LoadBitmap(hInstance, 'BMP_NAME');

    LoadBitmap загружает только картинку, без палитры. Если палитра у картинки отличается от системной, то ее надо устанавливать "вручную". Могут возникнуть проблемы, если на одной форме расположены две картинки с разными палитрами.

    procedure XLoadBitmap(Instance: THandle; BitmapName: PChar; var HB: HBitmap; var HP: Palette); 

    var 

      DC: HDC; 

      BI: PBitMapInfo; 

      Pal: PLogPalette; 

      I: Integer; 

      ResIdHandle: THandle; 

      ResDataHandle: THandle; 

      Bitmap: HBitmap; 

      C: HWnd; 

      OldPalette, Palette: HPalette; 

    begin 

      Bitmap := 0; 

      Palette := 0; 

      HB := 0; 

      HP := 0; 

    { Получить ресурс из модуля } 

      ResIDHandle := FindResource(Instance, BitmapName, RT_BITMAP); 

      if ResIDHandle <> 0 then 

      begin 

        ResDataHandle := LoadResource(Instance, ResIDHandle); 

        if ResDataHandle <> 0 then  

        begin 

          BI := LockResource(ResDataHandle); 

          if BI <> nil then 

          begin 

          { 256-цветный битмап? } 

            if BI^.bmiHeader.biBitCount = 8 then 

            begin 

            { Создать палитру } 

              GetMem(Pal, SizeOf(TLogPalette) + 256*SizeOf(TPaletteEntry)); 

              for I := 0 to 255 do with Pal^.palPalEntry[I] do 

              begin 

                peRed  := BI^.bmiColors[I].rgbRed; 

                peGreen:= BI^.bmiColors[I].rgbGreen; 

                peBlue := BI^.bmiColors[I].rgbBlue; 

                peFlags:= 0; 

              end

              Pal^.palNumEntries := 256; 

              Pal^.palVersion := $300; 

              Palette := CreatePalette(Pal^); 

              FreeMem(Pal, SizeOf(TLogPalette) + 256 * SizeOf(TPaletteEntry)); 

            { Привести цвета палитры в системные } 

              DC := CreateDC('Display', nil, nil, nil); 

              OldPalette := SelectPalette(DC, Palette, False); 

              UnrealizeObject(Palette); 

              RealizePalette(DC); 

            { Создать битмап } 

              BitMap:= CreateDIBitmap(DC, BI^.bmiHeader, CBM_INIT, 

                @PByteArray(BI)^[SizeOf(TBitMapInfo) + SizeOf(TRGBQuad) * 256 - 4], BI^, DIB_RGB_COLORS); 

            { Освободить ресурсы } 

              UnlockResource(ResDataHandle); 

              FreeResource(ResDataHandle); 

              SelectPalette(DC, OldPalette, False); 

              DeleteDC(DC); 

            end else  

            begin 

            { Не 256-цветный битмап } 

              UnlockResource(ResDataHandle); 

              FreeResource(ResDataHandle); 

              BitMap := LoadBitmap(Instance, BitmapName); 

            end

            HB := Bitmap; 

            HP := Palette; 

          end

        end

      end

    end

    procedure TForm1.FormCreate(Sender: TObject); 

    var 

      HB: HBitmap; 

      HP: HPalette; 

    begin 

      XLoadBitmap(hInstance, 'PHOTO', HB, HP); 

      Image1.Picture.Bitmap.Handle  := HB; 

      Image1.Picture.Bitmap.Palette := HP; 

    end


    10. Если я хочу рассылать EXE файл, созданный в Delphi, какие еще файлы нужно посылать с ним?

    Hикакие. Все компилируется в .EXE файл. Конечно, если вы разработали другие файлы (HLP, данные и т.д. ), или если вы используете VBX/OCX файлы, тогда вы должны распространять и их заодно. Если вы используете файлы VBX, то в поставку нужно также включать BIVBX11.DLL.

    Если приложение использует функции BDE, вы также должны включать Borland DataBase Engine.

    Полезные хитрости

    1. Может ли редактор текстов в Delphi вырезать и вставлять прямоугольные фрагменты текста?

    Конечно, может: Нажмите кроме Shift еще и Alt и режьте на здоровье. Alt можно сразу отпустить. Чтобы вернуться в старый режим, нужно выделить что-либо мышкой.


    2. Редактирование файлов SQL в Delphi IDE.

    Если вы в Delphi 2.0 IDE редактируете файл с расширением SQL, то, хотя это нигде не документировано, происходит автоматический Syntax Highlighting. Наибольший недостаток — не отслеживается конец комментария '*/'.

    В Delphi 3.0 комментарии отрабатываются нормально.


    3. Встроенный отладчик/дизассемблер.

    Если вы создадите в ключе

    HKEY_CURRENT_USER\Software\Borland\Delphi\2.0\Debugging

    строковое значение EnableCPU = "1", то после перезапуска среды у вас появится пункт меню View|CPU, которые вызывает появление простейшего отладчика/дизассемблера.

    Для Delphi 3.0 справедливо тоже самое (…\Delphi\3.0\Debugging, естественно), причем отладчик там по возможностям сравним с Turbo Debugger.

    Вопросы по Delphi 1.0

    Вопросы общего характера

    1. Какие существуют варианты поставки Delphi 1.0?

    Версия Delphi 1.0 имеет два варианта: Delphi Desktop и Delphi Client/Server.

    Версия Delphi 1.0 Desktop включает:

    • Среду разработки Delphi IDE

    • Механизм Borland Database Engine доступа к локальным данным Paradox и dBase, а также через ODBC

    • 16-разрядный Local Interbase

    • Печатную документацию

    Версия Delphi 1.0 Client/Server включает в себя все, что имеется в Delphi Desktop плюс:

    • SQL-Links 2.5, которые включают родные драйверы для Oracle, Sybase (MS SQL), Informix, и InterBase, и включают полные неограниченные права распространения к этим драйверам (что стоит $995 если это куплено отдельно);

    • Local InterBase Deployment Kit , $495[2];

    • ReportSmith/SQL, $300[3];

    • Средства поддержки групповой разработки — не доступно отдельно;

    • Visual Query Builder, который, как говорят, немного лучше, чем MSQuery, который входит в Microsoft Excel , Access , и т.д.). VQB также недоступен отдельно;

    • Исходные тексты библиотеки визуальных компонент, которые доступны отдельно за $100.

    • Дополнительно 2 тома документации.

    На данный момент версия Delphi 1.0 фирмой Borland отдельно не поставляется и имеется только в составе старших версий продукта.


    2. Какие форматы скомпилированных модулей можно получить в Delphi 1.0?

    Delphi может создавать EXE- и DLL-файлы  для Windows 3.1. Естественно, Вы может также создавать VBX, но для этого нужно знать соглашения по написанию DLL в формате VBX. Имеется информация о написании VBX для Borland Pascal for Windows, которая с небольшими изменениями подходит и для Delphi.

    Delphi не создает EXE-файлы для DOS.


    3. Есть ли проблемы в Delphi с русским языком?

    Что касается визуальных компонент, то все они, включая меню, допускают надписи (Caption) и ввод киррилицей; "горячие клавиши" тоже могут быть русские, например комбинация Alt-Ф для пункта меню &Файл (конечно, должен  быть включен драйвер русской клавиатуры). Для работы с таблицами нужно в утилите конфигурации BDE установить:

    1. В разделе драйверы для всех типов баз данных установить соответствующий драйвер языка (например Pdox ANSI Cyrillic).

    2. Если таблица (в частности, в формате Paradox) уже была создана с использованием другого драйвера языка, то ее можно перенастроить в DataBase Desktop на нужный драйвер.

    3. Для того, чтобы Database Desktop нормально 'видел' русские буквы, его настройки тоже необходимо немного подкорректировать.

    Русские буквы в среде:

    Поскольку используются разные версии Windows (Eng, Rus, Win-OS2, Win95, WinNT), способы могут быть как разными, так и общими (Windows есть Windows).

    Сначала опишем действия, необходимые для русификации Windows:

    1. Если вы собираетесь работать в OS/2, то желательно иметь английскую версию Windows 3.1 или OS/2 с Win-OS2 (в последнем случае вы не сможете запускать Windows без OS/2). Windows for WorkGroups здесь не подойдет, т.к. их сетевые функции под OS/2 работать не будут — для этого есть другие средства.

    2. Hесмотря на то, что вы собираетесь работать с англоязычной версией Windows найдите русские Windows и "вытащите" оттуда все шрифты *.FON, *.FOT, *.TTF.

    3. Каким-нибудь редактором шрифтов (напр FontoGrapher) скопируйте русские буквы с адреса 100 на адрес С0 — в этом случае один шрифт можно использовать и в английских и в русских Windows.

    4. Установите какой-либо русификатор — неплохо ведут себя CyrWin и ParaWin, причем для Win-OS2 предпочтительнее ParaWin. Устанавливать можно только русификатор без шрифтов, т.к. шрифты пойдут от русских Windows. Если вы не выполнили пункт 3, то TTF от русских Windows вам не помогут, и нужно будет ставить шрифты из комплекта русификатора.

    5. Замените все шрифты *.FON на шрифты из русских Windows.

    6. Подключите русские шрифты (Arial Cyr, Courier New Cyr, Times New Cyr).

    7. Добавьте в WIN.INI в секцию [FontSubstitutes] следующую строку: Arial=Arial Cyr или вместо Arial Cyr укажите русский шрифт сходный по начертанию (напр. для CyrWin это NTHelvetica/Cyrillic).

    Также на всякий случай можно добавить следующие подстановки:

    Helvetica=Arial Cyr (или NTHelvetica/Cyrillic)

    Одновременно закомментируйте строки, где упоминается английский шрифт Arial.

    В русских Windows можно совершенно безболезненно удалить (при помощи Control Panel/Fonts) шрифты, имеющие в названии окончание CE (напр. Arial CE) - это сербо-хорватские шрифты, которые вам вряд-ли когда понадобятся (русских букв там нет).

    Далее, возможные варианты работы с русскими буквами в редакторе Delphi.

    Нельзя в Windows выставлять TrueType fonts only — редактор использует только FixedFonts, в результате Delphi просто не будет работать.

    1. Комментарии и строки могут быть введены только в кодировке 1251 — причина, естественно в том, что русификатор позволит вводить русские буквы только в этой кодировке. Тексты, написанные в DOS (кодировка 866), будут отображаться как "мусор" на экране — редактор HЕ преобразует символы 866→1251. Однако если для редактора установить шрифт Terminal — тексты в 866 будут отображаться нормально, а ввести символы не удастся (опять же по причине ввода символов только в кодировке 1251). Уже готовый текст в 866 кодировке лучше преобразовать в 1251 с помощью одной из программ конвертации.

    2. Некоторые программные продукты при установке меняют фиксированные шрифты или даже удаляют их. Будьте внимательны при и после установки других программных продуктов, особенно редакторов текстов (WinWord, AmiPro…). Держите под рукой архив с русскими шрифтами *.FON, чтобы была возможность восстановить эти шрифты.

    3. Поэкспериментируйте со шрифтами редактора — разные шрифты имеют разное начертание, и разную скорость перерисовки. Выберите нужное для себя — либо скорость перерисовки, либо удобное начертание. Для локального InterBase нормальной русской кодировкой является Win1251 — имена доступных кодировок можно найти открыв любую DB InterBase (в т.ч. и локальную) и заглянув в системную таблицу RDB$CHARACTER_SETS.


    4. Какую модель данных использует Delphi?

    Delphi использует смешанную (mixed) модель памяти, но она очень похожа на модель large в C.С++:

    • Методы дальние (far)

    • Процедуры, описанные в интерфейсной части, дальние

    • Процедуры, используемые только в implementation по умолчанию ближние

    • Данные в Heap и все указатели вообще (включая экземпляры объектов) дальние

    • Глобальные переменные ближние (в сегменте DS)

    • Параметры процедур и локальные переменные ближние (в стеке)

    • Процедуры, объявленные far или export дальние

    • VMT дальние для новой модели классов и ближние для старой

    Эта схема используется в Borland Pascal долгое время.


    5. Можно ли использовать в приложении ресурсы, созданные в BPW 7.0?

    Все ресурсы, созданные в других приложениях, можно подключить и использовать  в Delphi с помощью директивы компилятора {$R ...} и процедур Windows API. Кроме того, меню из файла ресурсов и графические файлы *.BMP, *.ICO и *.WMF можно импортировать в приложение Delphi на этапе разработки. В настоящее время компанией Borland поставляется программный продукт RAD Pack for Delphi, где в состав поставки входит эксперт, позволяющий преобразовывать ресурсы из BP7.0 в формы Delphi.


    6. Возможно ли написать Screen Saver для Windows в Delphi?

    Для создания программы, работающей как Screen Saver:

    1. В проектном файле (*.dpr) напишите {$D SCRNSAVE <SaverName>} после uses; данная директива вставляет указанный текст (SCRNSAVE <SaverName>) в раздел описания модуля — в данном случае программы. Это главное, что необходимо для того, чтобы Windows распознал программу как Screen Saver.

    2. Hа главной форме выключите Border (BorderStyle=bsNone) и иконки. Установите свойства Left и Top =0, WindowState=wsMaximize.

    3. В обработчике события OnCreate, установите Application.OnMessage на процедуру деактивации Screen Saver. Установите Application.OnIdle на любую процедуру для рисования на экране.

    4. В обработчике OnCreate должна проверяться командная строка на наличие ключей /c и /s. Эти параметры определяют, нужно ли запускать сам Screen Saver или его конфигурацию (/c - конфигурация).

    5. Скомпилируйте программу и переименуйте из .exe в .scr ;  поместите файл в каталог Windows — Screen Saver должен появиться на панели управления (Control Panel).


    7. Как Delphi обрабатывает функции обратного вызова Windows (сallback)?

    Точно так же как C: вы можете получить указатель (far pointer) на вашу callback процедуру (не забыть при этом обьявить ее с директивой компилятора  {$F+}, либо спецификатором far) и передать этот указатель в Windows. Это все.

    Delphi и Visual Basic

    1. Есть ли в Delphi эквивалент массива элементов управления из Visual Basic?

    Hет. Компоненты Delphi не имеют свойства Index, подобное VB. Однако, имеются три основные причины, почему вы хотите использовать их в VB, и для каждой из них есть решение в Delphi.

    Причина 1. Вы хотите использовать один обработчик события для разных компонент на форме.

    Это просто. Все, что вы должны сделать — это выбрать тот же самый обработчик для каждой визуальной компоненты. Это лучше, чем в случае с массивом компонент, потому что вы можете одну и ту же процедуру-обработчик события присвоить различным типам компонент; например, кнопка на форме и пункт меню могут вызывать ту же самую функцию для обработки события OnClick.

    Причина 2. Вы хотите динамически создавать и уничтожать визуальные компоненты во время выполнения программы.

    Это также довольно просто в Delphi. Предположим, что на форме есть кнопка, и, каждый раз когда она нажимается, вы хотите создавать другую кнопку. Следующий пример показывается, как это делать:

    procedure TForm1.Button1Click(Sender: TObject);

    var

      NewButton: TButton;

    begin

      NewButton := TButton.Create(Self);

      NewButton.Parent := Self;

    end;

    Причина 3. Вам действительно требуется доступ к компонентам по номеру.

    Предположим, что вы решили написать игру вроде Реверси в Delphi. Вам нужно разместить 100 объектов TShape на форме, в виде квадрата 10×10. Конечно, размещать каждый элемент вручную на экране — задача трудоемкая и неинтересная, при этом, в декларации формы появляется 100 строк кода, которые, в общем-то, не нужны. Вместо этого можно завести массив вроде

    Board: array [1..10, 1..10] of TShape; Далее в программе нужно создать каждый из этих объекты, вызвав TShape.Create(FormXX); указать вручную начальные установки для них. Кроме того, в свойстве Parent каждого объекта из массива нужно указать ту панель (TPanel) или форму, на которой они располагаются. Это нужно для правильной перерисовки объектов.

    Если же вы не используете такие большие массивы, т.е., например хотите одинаково реагировать на нажатие 3-5 кнопок с незначительным различием для каждой из них, то можно использовать свойство Tag.


    2. Как использовать DLL, написанные в Delphi, например в Visual Basic?

    Допустим, вы написали на Delphi DLL и в нем объявили функцию

    function DataFileType(lpStr: PChar): Integer; export;

    begin

     StrPCopy(lpStr, oDataMap.VendorName);

     Result := StrLen(lpStr);

    end;

    в Visual Basic Вы должны подключить ее как:

    Declare Function DataFileType Lib "File.dll" (ByVal lpStr As String) As Integer

    а в программе на VB использовать следующим образом:

    nSize = DataFileType(lpVar)

    sVendorName = Mid$(lpVar, 1, nSize)


    3. Конвертация TBasicString (VBX) в string.

    Я использую VBX и испытываю проблемы с конвертацией TBasicString в string. Как это сделать?

    Существует две функции —

    • BStrPCopy — принимает Basic string и конвертирует в string

     SetBStr — принимает string и возвращает Basic string Действительно, ссылок на эти функции нет в документации, но имена этих функций можно "обнаружить" в VBXCTRL.DCU.

    Базы данных

    1. Ошибка инициализации BDE ($2C09).

    Когда я пытаюсь запустить приложение из Delphi, то получаю ошибку EDatabaseError и сообщение 'An error occurred while attempting to initialize the Borland Database Engine (Error $2C09)'

    Добавьте SHARE.EXE в AUTOEXEC.BAT или добавьте DEVICE=VSHARE.386 в раздел [386Enh] файла SYSTEM.INI и перезагрузитесь.


    2. Ошибка при загрузке языкового драйвера.

    У меня есть Quattro Pro 6.0 и IDAPI в сети. После установки Delphi и нового IDAPI поверх сетевого IDAPI при запуске Quattro Pro с другой машины я получаю ошибку 'Could not load Language Driver'.

    Добавьте раздел [Borland Language Drivers] в WIN.INI файл для указания каталога языкового драйвера. Пример:

    [Borland Language Drivers]

    LDPATH=C:\IDAPI\LANGDRV


    3. Что значит ошибка IDAPI $2C08?

    'Cannot load IDAPI01.DLL'. Убедитесь, что в файле WIN.INI правильно прописаны пути:

    [IDAPI]

    DLLPATH=C:\IDAPI

    CONFIGFILE01=C:\IDAPI\IDAPI.CFG


    4. Отличается ли локальный InterBase, встроенный в Delphi 1.0, от InterBase для других платформ, в частности, от InterBase для Windows NT?

    16-Разрядный Local InterBase не поддерживает:

    • функции, определяемые пользователем.

    • BLOB фильтры

    • сигнализатор событий (event alerters)

    • запись через журнал (Write Ahead Log (WAL)

    • тип данных массив (Array Datatype)

    • 'отключение' и 'включение' базы данных (database shutdown or restart)

    • ведение теневой базы данных (database shadowing)

    Все остальные функции поддерживаются, но структура хранения базы на диске не совпадает.

    Можно ли поставить локальный InterBase на файл-сервере и, таким образом, получить доступ к нему из многих приложений?

    Работать не будет. И не пытайтесь. Для этой цели вам нужен нормальный многопользовательский InterBase.


    5. Что насчет VBX-компонентов для работы с данными?

    Delphi поддерживает только VBX 1.0. Это значит что VBX для работы с данными 'не работает' с Delphi. В общем случае, все, что работает с Microsoft Visual C++, должно работать в Delphi. Кроме того, некоторые VBX достаточно хорошо написаны, так что их можно использовать в Delphi и без обращения к возможностям работы с данными.

    Object Pascal и Windows API

    1. Можно ли использовать OWL в Delphi?

    Можно, если уже существуют свои разработки с использованием OWL. Однако следует обратить внимание, не используются ли слова class, try, except и ряда других — в Delphi они являются зарезервированными. Если же опыта использования OWL не было, то вряд ли имеет смысл использование этой библиотеки — в Delphi существуют свои, более удобные средства для работы с Windows.

    Компиляция приложений, написанных на BP7 с использованием OWL.

    Компилятор Delphi способен компилировать приложения, написанные на OWL Borland Pascal 7.0. Для компиляции таких приложений в среде Delphi необходимо сделать следующее:

    1. Откройте главный файл приложения пунктом меню File|Open Project

    2. Добавьте в список используемых модулей uses модуль Messages в тех файлах, где использовался модуль WinTypes. Модуль Messages должен быть указан ранее модуля OWindows, ODialogs или любого другого OWL-модуля.

    3. При помощи пункта меню Options|Project Dialog добавьте путь x:\DELPHI\SOURCE\RTL70 к пути поиска модулей данного проекта. Этот каталог должен содержать файлы OWL, измененные для компиляции в Delphi.

    Если Вы забыли указать использование модуля Messages, то Вы при компиляции получите сообщение 'Unknown identifier'. Указание Messages после модуля OWindows вызовет сообщение 'Header does not match previous definition'.

    Идентификатор Result теперь используется особым образом внутри тела функций, поэтому их внимательно посмотрите, где этот идентификатор может встречаться. Недопустимо использование перекрывающихся диапазонов в операторе case.

    Изменилась реализация StrDispose. Если Вы распределяете память для строки при помощи GetMem, то освобождение этой области памяти при помощи StrDispose вызовет ошибку выполнения 'Runtime error 203'. Для распределения памяти для строк следует использовать функции StrAlloc или StrNew. Объекты OWL не ориентированы на обработку исключительных ситуаций, т.е. при возникновении таких ситуаций не происходит отката инициализации объектов. Мы не рекомендуем использование исключения в приложениях с OWL.

    Компилятор теперь производит поиск модулей, ресурсов и OBJ файлов по пути, указываемому в каталогах OPTIONS|Project. Путь 'EXE and TPU directory' больше не поддерживается. EXE-файлы могут создаваться в 'Output directory', а файлы DCU всегда создаются в том каталоге, где находятся соответствующие файлы PAS. Новые модули WinTypes и Messages могут быть скомпилированы BP7. Для этого не требуется включения дополнительных директив $IFDEF (зато там есть $IFDEF WIN32;).


    2. Как можно зааллокировать блоки памяти больше, чем 64 Кбайт?

    Используйте GlobalAlloc и GlobalLock из модуля WinProcs.


    3. GPF в ToolHelp.DLL в Win-OS/2 2.1.

    Поставьте FixPack, или сделайте Upgrade на OS/2 Warp 3 FullPack.


    4. Как получить из Clipboard текст большого размера?

    Да, стандартный метод TClipboard.AsText ограничивает текст размером строки - 255 байт. Для получения текста длиной более 255 байт можно использовать, например следующую процедуру:

    procedure GetLargeText: PChar

    var

      Buffer: PChar;

      MyHandle: THandle;

      TextLength : Integer;

    begin

      MyHandle := Clipboard.GetAsHandle(CF_TEXT);

      Buffer := GlobalLock(MyHandle);

      try

        Result := nil;

        if Buffer <> nil then

        begin

        { теперь у нас данные типа PChar -> мы можем работать

          с ними как с обычной Null-terminated строкой }

          TextLength := StrLen(Buffer);

          GetMem(Result, TextLength+1);

          StrCopy(Buffer, Result);

        end;

      finally

        GlobalUnlock(MyHandle);

      end;

    end;


    5. Проблемы Delphi с WinG.

    Я слышал, что у Delphi проблемы с WinG, однако кто-то их решил?

    Да, Майк Скотт (Mike Scott, 100140.2420@compuserve.com) даже написал коммерческий вариант VCL-компонентов, использующих WinG для Delphi:

    WinG Sprite Kit.

    Набор компонентов, осуществляющих доступ к WinG из Delphi. Включает компоненты TWinGCanvas для рисования TWinGDC и TWinGBitmap, TWinGSurface, которые можно поместить на форму и спрайтовый компонент, который можно помещать на них.

    $99 EEP

    Существует статья по использованию WinG в Borland Pascal — #5 Pascal Magazine. Обзор VCL WinG появится в новом Delphi Magazine. Отошлите письмо с вашим почтовым адресом на 70630.717@compuserve.com для получения бесплатной копии этих журналов (наверняка из России это не сработает.

    Компоненты и VCL

    1. Каковы ограничения на стандартные компоненты Delphi?

    Все компоненты, использующие TList для сохранения информации, имеют верхний предел 16368 единиц. Hапример, TTabControl может содержать до 16368 закладок и Delphi Component Palette может содержать до 16368 страниц.

    Многие из стандартных компонент Delphi являются надстройкой над стандартными управляющими элементами Windows. Windows 3.1 налагает свои собственные ограничения на эти компоненты. Hапример: TComboBox или TListbox могут содержать до 5440 единиц, а TMemo или TEdit (и соответствующие компоненты) — до 32k текста.

    Ресурсы Windows 3.1 ограничивают компонент TNotebook 570 страницами. (Трудно получить более 500 хендлов /handles/ окон в любом приложении Windows). Превышение этих границ вызывает ошибку или послужит причиной странного поведения Windows. 


    2. Предел буфера редактирования в 32K для TMemo.

    Почему в документации написано, что TMemo может редактировать тексты до 256К, а на деле получается не более 32К?

    В документации ошибка. 32К — это ограничение Windows. В Windows все стандартные редакторы используют общий буфер в 32К, компонент TMemo использует другую технику, позволяющую каждому TMemo иметь буфер в 32К. В Windows 95 эта проблема решена.


    3. Почему компонент TGauge так медленно работает (медленнее, чем VBX BGauge)?

    Компонент TGauge — просто пример, и ничего более. В нем отсутствует даже намек на оптимизацию перерисовок. Если вы посмотрите на код процедуры SetProgress, то увидите

    FCurValue := Value;

    Refresh;

    Это означает, что как только положение изменилось, весь компонент будет перерисован. Тем не менее, есть пути для того, чтобы сделать компонент более быстрым:

    1. Не присваивать Progress каждый раз (напр. 3000 раз), т.е. делать обновление менее часто

    2. Проверять, действительно ли позиция на экране изменится. Например, в SetProgress сделать следующее:

    if Abs(FCurValue-FLastDrawn) >= FDisplayDelta then 

    begin

      Refresh;

      FastDrawn := FCurValue;

    end;

    где FDisplayDelta что-то вроде (FMaxValue-FMinValue) div Width (идеально было-бы учитывать реальный размер экрана.

    3. Обновлять только часть индикатора, которая действительно меняется. Процедура Refresh стирает и перерисовывает весь компонент. Можно сделать вызов InvalidateRect (Windows API) и вызвать Update.

    4. Если вы сделали как в пункте 3, вы дополнительно можете оптимизировать процедуру перерисовки, чтобы не перерисовывать неизменяющийся участок.

    Разное

    1. Распространение приложений Delphi, использующих Local InterBase.

    Для распространения таких приложений, в соответствии с лицензионным соглашением вы должны перенести следующие файлы:

    Главные модули (X:\IBLOCAL\BIN)

    DSQL.DLL

    FILEIO.DLL

    GDS.DLL

    GBAK.DLL

    INTL.DLL

    IUTLS.DLL

    JRD.DLL

    REMOTE.DLL

    STACK.DLL

    Сообщения, лицензионные файлы и т.п. (X:\IBLOCAL)

    INTERBASE.MSG

    ISC4.GDB

    ISC_LIC.DAT

    Утилиты (если они необходимы)

    WISQL.EXE

    WISQL.HLP

    SQLREF.HLP

    IBMGR.EXE

    SVRMGR.HLP

    COMDIAG.EXE

    COMDIAG.INI

    COMDIAG.HLP

    BLINT04.HLP

    После копирования этих файлов необходимо проделать следующие операции:

    1. Добавить в AUTOEXEC.BAT в команду PATH  X:\IBLOCAL\BIN

    2. Там же сделать SET INTERBASE=X:\IBLOCAL

    3. В WIN.INI создать секцию

    [Interbase]

    RootDirectory=X:\IBLOCAL

    Естественно, если путь к локальному InterBase отличается от IBLOCAL, то вы должны изменить его на нужный.

    Для распространения BDE вы должны передать пользователю две инсталляционные дискеты с редистрибутивным BDE (на CD-ROM каталог REDIST\BDE).

    Для установки BDE вручную скопируйте содержимое каталога IDAPI (с подкаталогом языковых драйверов), и создайте в WIN.INI следующие секции:

    [IDAPI]

    DLLPATH=X:\IDAPI

    CONFIGFILE01=X:\IDAPI\IDAPI.CFG

    [Borland Language Drivers]

    LDPath=X:\IDAPI\LANGDRV

    После этого ненужные файлы (неиспользуемых языковых драйверов, поддержку ODBC…) можно удалить с диска, или временно перенести, убедиться в работоспособности приложения, и удалить.

    Возникает неясность — как добавить драйвер INTRBASE в IDAPI.CFG? Получается, что его необходимо переносить, и затем настраивать псевдонимы на новые каталоги. Иначе драйвер локального InterBase не попадет в IDAPI.CFG.ы

    Вопросы по Delphi 2.0

    Что нового в Delphi 2.0 по сравнения с Delphi 1.0?

    Выпущенная в феврале 1995 года версия Delphi 1.0 стала первым инструментом для Windows, комбинирующим оптимизирующий компилятор, механизмы визуальной разработки Two-Way-Tools и масштабируемую архитектуру обработки баз данных.  Сегодня сотни компаний по всему миру заявляют о многократной окупаемости их инвестиций в информационые системы, построенные с применением Delphi в качестве основного инструмента. Borland Delphi 2.0 полностью поддерживает все особенности новых операционных систем Windows 95 и Windows NT. Новый 32-разрядный оптимизирующий компилятор позволяет увеличить производительность разрабатываемых систем на 300-400 процентов при том, что генерируемый в результате код выполняется в 15-50 раз быстрее, чем в системах на базе P-код интерпретаторов. Новые объектно-ориентированные средства, предназначенные для разработки в архитектуре клиент-сервер, включают централизованное хранилище объектов — Object Repository и механизм визуального наследования форм — Visual Form Inheritance.  "Всего за один год Delphi был принят на вооружение заказчиками в силу уникальных возможностей этого продукта, сочетающего высокопроизводительную технологию компиляции с единой средой визуального программирования", — говорит вице-президент Borland по маркетингу (Product Marketing and Management) Ричард Горман (Richard Gorman). "С выпуском новых  версий мы расширяем рынок Delphi на всем спектре desktop, сетевых и клиент-серверных инструментов".

    Как и в любой новой версии продукта, по сравнению с предыдущей, появилось много изменений.

    1. Изменения в компиляторе и RTL

     1. Новые типы данных:

      • строки и символы поддерживающие произвольную длину и

      • кодировку UniCode.

      • вариантные структуры для работы с OLE Automation

      • тип Currency — 8-байтное число с плавающей точкой

     2. Переменные типа Integer и Cardinal теперь 32-битные (4 байта). Для работы с двухбайтовыми целыми числами необходимо использовать типы SmallInt и Word.

     3. Генерация 32-битного кода с оптимизацией циклов, передачей параметров через регистры, и т.п.

    2. Новые компоненты:

     1. набор компонент, свойственных интерфейсу Windows95

     2. компоненты OLE Automation

     3. новый DBGrid, позволяющий определять атрибуты столбцов.

    3. Изменения в работе с БД

     1. локальная фильтрация записей для TTable и TQuery

     2. поддержка lookup у TField

     3. SQL-монитор, отслеживающий выполнение SQL-операций

     4. Модуль Данных (DataModule), для централизованного хранения и использования компонент доступа к базам данных

    4. Изменения в среде разработчика (IDE)

     1. хранилище объектов (Object Repository) — для хранения проектов, форм, модулей данных и др.

     2. визуальное наследование форм

     3. визуальное связывание форм

     4. Database Explorer

     5. Редактор полей таблиц в стиле drag-n-drop

    5. Изменения в Borland Database Engine

     1. полностью 32-разрядная библиотека доступа к данным

     2. новое ядро SQL-запросов

     3. расширенные возможности SQL Links

     4. транзакции и вложенные запросы для локальных форматов данных (dBase и Paradox)

    Вопросы общего характера

    1. Какие существуют варианты поставки Delphi 2.0?

    Выпущенная 4 февраля 1996 года серия продуктов Delphi 2.0 включает три версии, каждая из которых разработана с учетом различного уровня разработчиков и решаемых ими задач:

    • Desktop — для создания автономных программ или для начинающих программистов

    • Developer — для профессиональных разработчиков, ориентированных на сетевую архитектуру

    • Client/Server Suite — для создания систем в архитектуре клиент-сервер Все версии Delphi 2.0 естественно сочетают высокопроизводительный 32-разрядный компилятор, масштабируемые инструменты доступа к базам данных и расширяемую библиотеку "drag-and-drop" компонент в составе объектно-ориентированной среды визуальной разработки.

    Состав версий.

    Все версии Delphi 2.0 обладают открытой архитектурой, полностью поддерживающей такие технологии, как OLE  server, Microsoft OLE Controls (OCX), ODBC, а также Microsoft's  Remote Automation и ожидаемую Network OLE (ActiveX). Все версии Delphi 2.0 также предоставляют разработчикам поддержку новых особенностей и интерфейсов прикладного программирования (API) Windows 95 и Windows NT - многопоточности (threads), Unicode, MAPI и др. Для облегчения перехода разработчиков из 16-разрядного в 32-разрядное операционное окружение каждая версия Delphi 2.0 включает 16-разрядную версию Delphi 1.02 for Windows.

    Delphi Desktop 2.0

    Delphi Desktop 2.0 наиболее всего соответствует Delphi 1.0 for Windows и предназначен для начинающих Windows-прогрммистов и индивидуальных разработчиков. Некоторые особенности Delphi Desktop 2.0:

    • оптимизирующий 32-разрядный компилятор, увеличивающий производительность существующих приложений на 300-400% (относительно Delphi 1.0).

    • среда разработки IDE с интерфейсом в стиле Windows95

    • расширяемая объектная архитектура компонент

    • визуальное наследование форм

    • визуальное связывание форм и компонентов, размещенных на различных формах

    • 32-разрядный Borland Database Engine для доступа к БД формата dBase и Paradox, обеспечивает ряд расширений языка запросов SQL, транзакции для локальных форматов данных

    • Data Modules (модули данных), позволяющие использовать одни и те же таблицы, запросы и др. источники данных и компоненты из многих форм приложения

    • Database Explorer — инструмент разработки и модификации структур и содержимого баз данных в стиле Windows Explorer.

    • фильтры для таблиц и запросов, развитые Lookup-списки.

    • расширенный Grid-компонент с настраиваемыми атрибутами столбцов и "выпадающими" списками

    • компонент Quick Report, позволяющий легко создавать встроенные отчеты без использования ReportSmith.

    • тип данных currency (деньги), увеличивающий точность финансовых вычислений

    • длинные строки и структуры данных (до 2ГБ)

    • 16-разрядная версия Delphi 1.02

    • полная документация в 5-и томах (более 1100 страниц) включая полное описание языка.

    Delphi Developer 2.0

    Delphi Developer 2.0 ориентирован на поддержку профессиональных разработчиков многопользовательских (сетевых) приложений. Версия Developer по сравнению с Desktop имеет следующие расширения:

    • хранилище объектов (Object Repository), поддерживающее создание и совместное использование форм, модулей данных и других объектов.

    • масштабируемый словарь данных (Data Dictionary), содержащий расширенные атрибуты полей (столбцов), квлючая пределы величин, маски редактирования и отображения, параметры шрифтов и т.п.

    • низкоуровневая поддержка Borland Database Engine, включая справочные файлы

    • Multi-Object Grid для максимально гибкого прдставления информации в приложениях, работающих с базами данных

    • расширенный набор примеров компонент и дополнительные OCX

    • дополнительные эксперты, среди которых Installation/Deployment Expert для создания рсапространяемых приложений (дистрибутивов)

    • WinSight32

    • расширенный Open Tools API — набор открытых интерфейсов для интеграции с внешними инструментами (CASE's, Transaction Process Monitor's и др.), с помощью которых разработчик может создавать и встраивать в среду (IDE) Delphi свои редакторы компонент и их свойств, эксперты и другие инструменты

    • интерфейс к средствам групповой разработки (требует Intersolv PVCS 5.2 или выше)

    • локальный однопользовательский сервер InterBase для разработки масштабируемых приложений на отдельном компьютере

    • 32-разрядный генератор отчетов ReportSmith 3.0 с расширенными возможностями интеграции с Delphi-приложениями

    • новая библиотека математических, статистических и бизнес-функций исходные тексты библиотеки компонент VCL32 (32-bit Visual Components Library)

    • 8 томов документации и справочных руководств общим объеком свыше 3000 страниц

    Delphi Client/Server Suite 2.0

    Delphi Client/Server Suite 2.0 ориентирован на организации, разрабатывающие корпоративные системы, предназначенные для работы с данными, хранимыми на серверах БД Oracle, Sybase, InterBase, Informix, MS SQL Server, DB/2; сочетает высокопроизводительный клиентский инструментарий и широкий спектр средств работы с серверами БД. Версия Client/Server Suite по сравнению с Developer имеет следующие расширения:

    • SQL Explorer в стиле Windows Explorer, ориентированный на обработку метаданных серверных БД (доменов, триггеров, представлений, хранимых процедур и т.п.)

    • SQL Monitor, предназначенный для тестирования, отладки и настройки SQL-запросов для повышения качества и производительности их выполнения

    • Cached Updates (буферизированное обновление) обеспечивает более эффективную обработку транзакций в клиент/серверном окружении.

    • неограниченное использование высокопроизводительных драйверов SQL Links для доступа к серверным БД Oracle, Sybase, InterBase, Informix и SQL Server (лицензия на распространение SQL Links).

    • SQL-сервер Borland InterBase 4.1 для Windows NT с лицензией на 2-х пользователей.

    • ReportSmith 3.0 SQL Edition, предназначенный для построения отчетов при работе в клиент/серверном окружении.

    • Визуальный конструктор запросов Visual Query Builder

    • DataPump Expert — средство переноса/миграции данных для масштабирования приложений. Интегрированная в среду разработчика система контроля версий Intersolv PVCS.

    • 12 томов документации и справочных руководств, общим объемом свыше 3500 страниц.


    2. Какие форматы скомпилированных модулей можно получить в Delphi 2.0?

    Delphi может создавать EXE- и DLL-файлы  для Win 32. Естественно, Вы может также создавать OCX, но для этого нужно знать соглашения по написанию DLL в формате OCX.

    Delphi может также создавать 32-разрядные консольные приложения для работы под Win 32.


    3. Какую модель данных использует Delphi?

    Delphi 2.0 использует так называемую плавающую модель памяти (FLOAT), которая принята в Win 32. Отличительной особенностью данной модели памяти является линейная 32-разрядная адресация всего адресного пространства, которое может иметь соответственно размер до 4 Гбайт. При этом все указатели, адреса процедур, указатели на VMT также адресуются через 32-разрядные регистры.


    4. Delphi 2.0 может создавать 16-разрядные приложения?

    Delphi 2.0 — это полностью 32-разрядный продукт, который создает приложения, функционирующие под Windows 95 и Windows NT. Но в комплект поставки Delphi 2.0 также входит и текущая 16-разрядная версия Delphi (версия 1.0), предназначенная для создания приложения, работающих под Windows 3.1 (а также Windows 3.11 и др.). Те, разработчики, которые не используют новые специфичные характеристики 32-разрядных операционных систем погут перекомпилировать свои приложения с использованием 16-разрядной версии Delphi.


    5. Насколько трудно перенести существующее приложение Delphi в Delphi 2.0?

    В большинстве случаев разработчику достаточно просто перекомпилировать свое приложение с помощью нового 32-разрядного оптимизирующего компилятора и сразу же использовать преимущества 32-разрядного кода, что вызывает увеличение производительности до 300-400% под Windows 95 и Windows NT. Тем разработчикам, которые использовали низкоуровневый код, использующий 16-разрядную сегментную архитектуру Windows 3.1, больше не поддерживаемую Windows 95 придется внести соответствующие изменения в коде. Если приложение использует дополнительные компоненты и библиотеки третьих фирм, то следует обратиться к фирме-производителю для получения 32-разрядных версий этих компонентов и библиотек.


    6. Каким образом разработчик может использовать новые характеристики Windows 95?

    Delphi 2.0 включает в себя множество новых компонентов для поддержи новых специфичных характеристик Windows 95, таких как элементы оформления интерфейса пользователя (включая редактирование текстов в формате RTF), многостраничный диалог и прогресс-индикатор в стиле Windows 95, OLE controls (OCX) и др. Разработчику достаточно просто добавить эти компоненты в свои приложения из палитры компонентов, как и любой другой компонент Delphi. В большинстве случаев, Delphi 2.0 будет автоматически поддерживать новые возможности, например такие как длинные имена файлов, новые диалоги и стили и др. В дополнение ко всему, ввиду того, что Delphi 2.0 компилятор непосредственно в исполняемые коды процессора, разработчики сразу же получают доступ ко всему API Windows 95, включая мультитрединг, строки в формате Unicode, MAPI и др.


    7. Delphi 2.0 сертифицировано как продукт под Windows 95?

    Да. Delphi 2.0 удовлетворил всем требованиям для сертификации как продукт Windows 95. Кроме того, Delphi 2.0 облегчает разработчикам создание приложений, которые могут быть сертифицированы для использования под Windows 95.


    8. Delphi 2.0 поддерживает Windows NT?

    Да, Вы можете работать с Delphi 2.0 как под Windows 95, так и под Windows NT и, соответственно, создавать приложения, которые будут работать под обеими платформами. При этом имейте ввиду, что не все функции Win 32 API могут работать на обоих платформах, например, в Windows 95 не реализованы сервисы и др. системные функции — но это уже проблема не Borland, а Microsoft.


    9. Delphi 2.0 поддерживает OLE controls (OCX) и OLE automation?

    Да. Delphi 2.0 полностью поддерживает OCX и OLE automation. Разработчики могут инсталлировать OLE controls на соответствующие страницы палитры компонентов или использовать уже поставляемые с Delphi 2.0. Также, разработчик может использовать OLE automation для того, чтобы создавать приложения, которые управляют другими приложениями, такими как Microsoft Word, Excel, Lotus 1-2-3, Borland C++, Paradox и др. OLE automation в Delphi полностью совместима дальнейшем развитием OLE — Network OLE а также с технологией удаленной автоматизации (remote automation), включенной в VB4, используя все преимущества оптимизирующего компилятора.


    10. Delphi 2.0 поддерживает in-process (DLL) или out-of-process (EXE) серверы в OLE automation?

    Delphi 2.0 полностью поддерживает эти два типа локальных серверов OLE automation.


    11. Delphi 2.0 поддерживает другие виды взаимодействия между приложениями?

    Да. Delphi взаимодействует с некоторыми мониторами выпонения транзакций (transaction processing — TP), включая Novell Tuxedo, TransArc Encina, CICS и др. Как правило, эти продукты оформлены в виде DLL и могут вызываться непосредственно из приложения. Кроме того, Borland состоит в Object Management Group (OMG) и планирует в будущем обеспечить поддержку CORBA.


    12. Delphi 2.0 поддерживает мультитрединг?

    Да. Так как Delphi компилирует непосредственно в коды команд процессора, приложению написанному на Delphi доступны все возможности API Windows 95 и Windows NT. Библиотека Визуальных Комонентов (VCL) также включает объект TThread для создания надежных приложений.


    13. Delphi 2.0 совместима с Network OLE?

    Да. Так как Delphi 2.0 полностью поддерживает OLE automation как серверы, так и контроллеры, данный вариант OLE полностью совместима в будущем с Network OLE. Также Delphi полностью поддерживает технологию удаленной автоматизации, включенной в VB 4.0 со всеми дополнительными преимуществами оптимизирующего компилятора.


    14. Существует ли upgrade Delphi 2.0?

    Да. Borland предлагает специальные цены для тех заказчиков, которые являются легальными пользователями Delphi версии 1.0.


    15. Насколько успешны продажи Delphi на сегодняшний день?

    Продажи Delphi и Delphi Client/Server на данный момент даже превосходят прогнозы, сделанные фирмой Borland. Delphi играет значительную роль как в разработке отдельных, независимых приложений, так и в разработке приложений по технологии клиент-сервер. По результатам опросов и тестов Delphi присуждено множество наград, в т.ч.:

    • BYTE Best Technology of Comdex 1994 for best development/system software

    • PC Week Labs Analyst Choice Award

    • PC Magazine Top Ten Selling Products

    • PC Magazine Technical Excellence Award

    • PC Magazine Product of the year for 1995

    • Computer Daily News (Australia), Top Ten Selling Products

    • PC Magazine (UK) Grey Matter Award for the number-one selling software product

    • PC World (Spain) Product of the Year for programming languages

    • PC/Computing Excellence Award

    • DBMS Reader's choice award

    • Best of LAN Times

    • Windows Tech Journal Star Tech award

    • Ziff-Davis Cannes Software Excellence Award, Overall Technical Excellence

    • Ziff-Davis Cannes Software Excellence Award, Languages and Tools

    • Visual Basic Programmer's Journal Editor's Choice


    16. Насколько полно Borland предлагает стратегические решения тем компаниям, которым требуется дополнительная помощь в разработке клиент-серверных приложений?

    В первом квартале 1996, Borland представил новую программу Premier Partner VAR, где главный акцент делается на решения в рамках технологии клиент-сервер. Эта программа будет играть важную роль в дополнении комплекта Delphi Client/Server Suite средствами, предназначенными для основных вариантов разработок по технологии клиент-сервер.


    17. В чем заключаются преимущества 32-разрядного компилятора в родной код микропроцессора?

    Используя новый оптимизирующий компилятор в 32-разрядный код Delphi 2.0 генерирует приложения, превосходящие по своим качествам интерпретаторы p-кода, такие как Visual Basic и PowerBuilder. На данный момент приложения, произведенные с помощью Delphi 1.0 выполняются примерно в 10-20 раз быстрее интерпретаторов p-кода. Предполагается дополнительное увеличение этого показателея в 32-разрядной версии. Новый оптимизирующий 32-разрядный компилятор в Delphi 2.0 использует один и тот же генератор кода, что и Borland C++ и включает множество видов оптимизации, таких как оптимизация использования регистров, оценка общих частей выражений оптимизация использования переменных и генерация кода, оптимированного для выполнения на процессоре Pentium для получения более компактного и быстрого кода. Приложения перекомпилированные во второй версии меньше по размерам и работают быстрее, чем раньше. Ввиду нового кодогенератора появились возможности не только использовать, но и создавать OBJ-файлы для более легкого переноса кода между Delphi и C/C++. Кроме того, сейчас компилятор выдает более информационные сообщения об ошибках, равно как и предупреждения и советы о некорректном коде, неиспользованных или используемых до инициализации переменных и др.


    18. Насколько быстрее работают приложения, скомпилированные новым компилятором?

    Тестовые испытания показали, что код, полученный при помощи Delphi 2.0 работает в среднем на 300-400% быстрее, чем 16-разрядные приложения. Это означает, что новые приложения будут работать в 15-50 раз быстрее, чем интерпретаторы p-кода. Например, тесты Sieve (что такое - не знаю) показали, что Delphi 2.0 работают в 15 раз быстрее, чем VB 3.0 и в 815 раз быстрее, чем PowerBuilder 4.0.

    Ниже приводятся результаты тестирования 16- и 32-разрядных версий Delphi. Все тесты выполнялись на Gateway 2000 V66 (66MHz, процессор 486) с 16Mb памяти. 16-разрядные тесты выполнялись под Windows 3.1. Новые 32-разрядные тесты были выполнены с использованием пре-релиза Delphi 2.0.

    Большее число означает большую производительность  

    Power Builder  Visual Basic  Delphi 1.0  Delphi 2.0
    Sieve  0.22  11.95  52.77  179.37 
    Whetstone  0.04  1.41  4.70  15.53
    File read  0.05  0.42  0.74  2.89 
    File write  0.05  0.33  1.75  5.28 

    19. Какой вид коллективной работы над проектом поддерживает Delphi 2.0?

    Delphi Client/Server Suite 2.0 использует Open Tools API для тесной интеграции с системой контроля версий Intersolv PVCS (сама система входит в комплект поставки) для работой с файлами. Delphi Developer 2.0 также имеет интерфейс к Intersolv PVCS, но не включает саму систему PVCS. Благодаря Open Tools API, разработчики могут самостоятельно подключать другие системы контроля версий, такие как MKS Source Integrity, Microsoft Source Safe и др.


    20. Насколько Delphi Client/Server Suite 2.0 сравним с PowerBuilder?

    Delphi Client/Server Suite 2.0 включает в поставку большой диапазон средств, которые имеются в PowerBuilder и предлагают профессиональным разработчикам приложений по технологии клиент-сервер все, что им требуется для построения надежных, мощных приложений по технологии клиент-сервер. Кроме того, предлагаются дополнительные средства, такие как SQL Explorer для просмотра мета-данных на сервере, таких как сохраненные процедуры, триггеры и обработчики событий, SQL Monitor для тестирования и отладки SQL запросов, поддержка командной работы над проектом со встроенным PVCS, расширенный Open Tools API для интеграции с CASE-средствами и версия  InterBase под NT с 2-мя пользовательскими лицензиями. Также новый Borland Database Engine поддерживает буферизированное обновление, что существенно облегчает создание высокопроизводительных многозадачных приложений. Delphi остается единственным стредством, совмещающим преимущества оптимизирующего компилятора в родной код, визуального проектирования и технологии масштабируемой работы с базами данных.

    Хотя компания PowerSoft и опубликовала планы реализации генератора C кода с возможностью последующей его компиляции в середине 1996 года, он не сможет по производительности, простоте отладки и интеграции в среде превзойти Delphi, который основан на 10-летней технологии производства компиляторов.


    21. Насколько Delphi 2.0 сравним с Visual Basic 4.0?

    Новый 32-разрядный оптимизирующий компилятор в Delphi 2.0 обеспечивает даже более высокие показатели, чем интерпретатор p-кода в VB 4.0. Кроме того, Delphi 2.0 предлагает более полный доступ к возможностям Windows 95 и NT вместе с поддержкой мульти-трединга и строк Unicode в дополнение к OLE Controls (OCX) и OLE automation. На самом деле, Delphi 2.0 может использовать все преимущества технологии удаленной автоматизации, включенной в VB 4.0 с дополнительным увеличением в скорости.

    Delphi 2.0 превосходит VB также в поддержке дополнительных средств технологии клиент-сервер, таких как Database Explorer, Object Repository и Data Dictionary. Также Delphi 2.0 предлаагет много нововведений, таких как визуальное наследование форм и др.

    Delphi остается единственным стредством, совмещающим преимущества оптимизирующего компилятора в родной код, визуального проектирования и технологии масштабируемой работы с базами данных. VB 4.0 получил очень среднюю оценку от заказчиков как просто некий 32-разрядный вариант предыдущей версии. VB 4.0 так и не использует основные возможности, такие как компилятор родного кода и настоящее объектно-ориентированное программирование.

    Базы данных

    1. В чем заключаются преимущества нового 32-разрядного Borland Database Engine?

    Новый 32-разрядный Borland Database Engine включает полностью новое ядро запросов, которое было оптимизировано для работы как с удаленными SQL-серверами, так и с локальными данными. 32-разрядный Borland Database Engine использует все преимущества 32-разрядного адресного пространства и асинхронный 32-разрядный ввод/вывод для повышения характеристик. Он также включает новые виды оптимизации, специфичные для конкретных серверов и форматов локальных баз данных. Например, новые 32-разрядные SQL-линки включают многие виды оптимизации для Oracle, Sybase, Informix и InterBase. Новый BDE поддерживает буферизированное обновление для улучшения характеристик выполенний транзакций в приложениях, интенсивно работающих с данными на сервере без требуемого ранее блокировния используемых ресурсов на последнем. Кроме того, дополнительно поодерживаются транзакции на локальных данных и новое ядро запросов более полно реализует стандарт ANSI SQL-92 DML compliance. Новый Borland Database Engine поддерживает Data Dictionary для хранения расширенных аттрибутов данных, таких как минимальное и максимальное значения, маски для редактирования и вывода и др. Delphi также включает новую 32-разрядную версию локального сервера InterBase Server для более полной возможности масштабирования приложений в возможность использования любого другого ANSI SQL 92 сервера.


    2. Что такое сервер InterBase?

    InterBase это высокопроизводительный SQL сервер фирмы Borland выпускаемый под различные платформы. InterBase доступен для более чем 15 операционных систем, включая: Windows 3.1, Windows 95, Windows NT, NetWare, SCO, Sun OS, Sun Solaris, HP-UX, IBM AIX, SGI IRIX, и множество других Unix платформ.


    3. Что такое локальный (Local) InterBase?

    Это однопользовательская версия InterBase, включенная в Delphi Developer 2.0 и Delphi Client/Server Suite 2.0. Local InterBase дает возможность разработчикам тестировать свои приложения без привлечения настоящего SQL-сервера, но в то же время используя стандарт ANSI 92 SQL. Локальные InterBase имеет все те же основные возможности, что и многопользовательская версия InterBase доступная под NT и Unix, включая управление транзакциями, сохраненные процедуры, триггеры и обработчики событий.

    С помощью локального InterBase разработчики на Delphi получают возможность тестировать свои настоящие клиент-серверные приложения на одном рабочем месте. Это означает возможность работы на laptop-е во время поездок либо использование баз данных, которые очень быстро меняют свою структуру. В отличие от других серверов, InterBase имеет один и тот же API для всех 15, так что приложение, работающее с локальным InterBase будет работать и с любой Unix или NT версией InterBase сервера без каких-либо модификаций.


    4. В 32-разрядной версии локального Local InterBase были сделаны какие-либо улучшения?

    Да, новая 32-разрядная версия локального InterBase предлагает значительное улучшение характеристик 16-разрядной версии. Также предлагается исключительный интерфйес Windows 95 GUI, включая 32-разрядные средства — Server Manager и Interactive SQL, и полную on-line документацию.


    5. Как следует распространять приложения Delphi, использующие InterBase?

    Используя Delphi Client/Server Suite 2.0, разработчик может проектировать и тестировать свое приложениe, используя связку Delphi/InterBase на одном рабочем месте. После того, как приложение закончено, просто покупается и инсталлируется требуемая версия InterBase с нужным количеством клиентских лицензий. Данные просто переносятся на сервер, после чего приложение готово к работе. Характеристики InterBase аналогичны для всех платформ - Windows 95, NT и Unix, так что не имеет значения, с какой операционной средой работает сервер. Кроме того, Delphi Developer 2.0 и Delphi Client/Server Suite 2.0 включают InstallShield Express — средство для создания инсталляторов.


    6. Насколько InterBase сравним с Watcom SQL?

    Характеристики InterBase и Watcom SQL во многом схожи, например в наличии хранимых процедур, триггеров и полного котроля за транзакциями. Однако, InterBase полностью реализует ANSI SQL 92, поддерживает событийную систему на сервере для программирования событийно — управляемых приложений и одновременную модель работы для многопользовательского доступа. Watcom не реализует полностью ANSI 92, не поддерживает события на сервере и имеет полность другое ядро, чем Sybase. Аналогично, InterBase масштабирует с 16-разрядной операционной системы Windows в Unix и имеет один и тот же интерфейс [API] для всех платформ. Watcom не имеет Unix-версий и имеет различные API для всех версий Sybase. Следовательно, если приложение разрабатывается с использованием Watcom API для Windows 3.1, Windows 95, или NT, они должны полностью переписывать свое приложение при изменении целевой платформы. InterBase обеспечивает полностью переноимые, масштабируемые решения для разработчиков клиент-серверных приложений. Наконец, в отличие от блокироки сираниц в Sybase System 11, InterBase предлагает блокировку запись в соответствии с архитектурой множественных поколений записей.


    7. Что следует предпочесть разработчику в Delphi: SQL RDBMS, подобный InterBase или базу данных PC LAN, подобно dBase или Paradox?

    Для небольших приложений, примерно до 12 одновременно работающих пользователей, использование БД PC LAN таких как dBase или Paradox даст максимально высокую производительность. SQL RDBMS, аналогичные InterBase, проявляют свои преимущества при более чем 10 пользователях и при высоких требованиях к одновременной многопользовательской работе, а также высокой степени сохранности данных. Клиент-серверные приложения позовляют также манипулировать данными больших размеров (20Mb — 20Gb) которые в другом случае вызвали бы слишком медленную работу вследствие взаимного блокирования файлов на файл-сервере и высокого сетевого траффика.


    8. Почему я не могу создать в Local InterBase таблицу с кодовой страницей, отличной от страницы по умолчанию?

    Этот случай наблюдается, когда полный путь к каталогу, где находится Local InterBase 32 содержит хотя бы один символ пробела (что предлагается по умолчанию). Ошибка официвльно признана Borland и будет устранена в ближайших следующих версиях Delphi 2.0, а тем, кто приобрел текущий вариант Delphi следует перенести IB в другой каталог (например, C:\INTRBASE) и перенастроить пути, либо полностью переставить Delphi с учетом данного требования.

    Object Pascal и Windows API

    1. Что такое Open Tools API? Насколько он улучшен?

    Delphi разработан с учетом Open Tools API, который предоставляет средства интеграции средств третьих фирм, таких как систем контроля версий (Version Control System — VCS), CASE-средств, экспертов и т.д. В версии Delphi 2.0, Open Tools API был расширен с целью увеличить уровень интеграции в отношении работы с файлами, редактором и др. Эта технология идет дальше Microsoft's Source Code Control (SCC) API для того, чтобы обеспечить более общие возможности интеграции. В данное время Borland ведет работу с большим количеством дополнительных третьих фирм над улучшением взаимодействия с ведущими средствами CASE-проектирования, таких как Popkin System Architect, Sybase S-Designor, CSA SilverRun, LBMS и FMI Select Tools Enterprise и др.

    Разное

    1. Как ReportSmith 3.0 взаимодействует с Delphi?

    Новая 32-разрядная версия ReportSmith 3.0 предлагает более тесную взаимосвязь со средой Delphi и способна работать с любым источником данных, используемом в Delphi, таких как TQuery или TTable. ReportSmith поддерживает создание сложных запросов в среде клиент-сервер и способен функционировать с приложением любой сложности. Delphi 2.0 также включает набор компонетов TQuickReport для встраивания отчетов непосредственно вовнутрь приложения.


    2. Каким образом можно разделять Delphi 2 Object Repository между несколькими машинами?

    Для этого следует используя Regedit в Registry в секции

    HKEY_USERS\.Default\Software\Borland\Delphi\2.0\Repository

    завести строковую запись BaseDir и в ней указать путь к вашему репозитарию (в том силе и сетевой).

    Вопросы по Delphi 3.0

    Вопросы общего характера

    1. Какие существуют варианты поставки Delphi 3.0?

    Анонсировано три варианта поставки Delphi, каждый из которых предлагает комплект решений для разного уровня разработчиков и решаемых ими задач.

    • Standard

    • Professional

    • Client/Server Suite

    Все версии Delphi 3.0 включают в себя высокопроизводительный 32-разрядный оптимизирующий компилятор, масштабируемые средства доступа к данным, расширяемую библиотеку компонентов, объединенные единой объектно-ориентированной средой разработки.

    Состав версий.

    Все версии Delphi 3.0 обладают окрытой архитектурой, полностью поддерживают технологии Microsoft OLE Automation, OCX, ODBC, ActiveX. Компонентная модель COM/DCOM поддерживается на уровне компилятора. Компилятор позволяет вам иметь доступ ко всем ресурсам операционных систем, реализующих Win 32 (Windows 95 и Windows NT) и использовать все имеющиеся технологические стандарты - Unicode, MAPI, ISAPI, NSAPI. Как в версии 2.0, в поставку включается 16-разрядная версия Delphi 1.02.

    Delphi Standard 3.0

    Delphi Standard 3.0 ориентирован на разработчиков в основном отдельных приложений с использованием настольных баз данных. Основные характеристики варианта Standard:

    • Профессиональная среда разработки, включающая в себя полностью интегрированный отладчик и редактор

    • Интегрированный в среду 32-разрядный оптимизирующий компилятор

    • Возможность создания DLL и отдельных исполняемых EXE-файлов

    • Возможность создания очень легких EXE с использованием настраиваемой технологии pakeges.

    • Полный доступ к Win32 API, поддержка ActiveX, OLE, OLEDB, COM, DCOM, MAPI, ISAPI, NSAPI

    • Создание и отладка многопоточных приложений под Windows 95 / Windows NT

    • Наличие Delphi 1.0 для создания 16-разрядных приложений под Windows 3.1

    • Объектно-ориентированная расширяемая компонентная архитектура

    • Наличия репозитария объектов для хранения и повторного использования форм, модулей данных

    • Поддержка визуального наследования и визуального связывания форм для уменьшения размера вводимого кода и более простого управления

    • Полный набор новейших управляющих элементов Windows 95

    • Визуальная библиотека компонентов (VCL) с более чем 100 компонентами, доступными для повторного использования методом 'drag-and-drop'

    • Встроенный генератор отчетов, состоящий из компонентов, позволяющий создавать, просматривать и печатать отчеты без использования дополнительных внешних программ

    • Использование ActiveX

    • Поддержка COM и интерфейсов на уровне языка и компилятора

    • Полная поддержка серверов и контроллеров OLE Automation

    • Визуальное создание шаблонов для новых компонентов

    • Упрощение кодирование и завершение кода с помощью CodeTemplates wizard, CodeCompletion wizard и CodeParameter wizard

    • Быстрое вычисление выражений в ToolTip окошке для облегчения процесса отладки Отладка DLL для сокращений времени на разработку и отладку DLL

    • Поддержка механизма многих источников данных для быстрого доступа к данным в любой СУБД

    • Высокопроизводительные родные драйвера для доступа к данным MS Access, FoxPro, Paradox, dBase

    • Компоненты для работы с данными для построения высокопроизводительных приложений, работающих с базами данных

    • Увеличение скорости обмена с сервером за счет поддержки Cashed Updates

    • Выделение бизнес логики в отдельный модуль данных

    • Визуальное упраление базами данных с помощью Database Explorer

    Delphi Professional 3.0

    Delphi Professional 3.0 предназначается для разработчиков многопользовательских приложений. Данная версия включает в себя все, что имеется в Delphi Standard плюс:

    • Исходный код VCL и печатные материалы для более качественного создания собственных компонентов

    • Компоненты для построения графиков и диаграмм на основании данных, хранящихся в таблицах баз данных

    • Визуальное создание элементов ActiveX для приложений, ориентированных на Web

    • Визуальное создание элементов ActiveX для повышения степени повторного использования кода

    • Полная поддержка доступа к данным через ODBC

    • Обеспечение целостности данных с помощью Data Dictionary

    • Возможность создавать и тестировать приложения, работающие с Local InterBase (одна пользовательская лицензия)

    • Internet Solutions Pack для создание приложений, использующих ресурсы Web

    • Install Shield Express для создания профессиональных инсталляторов

    • Open Tools API для интеграции с вашими дополнительными утилитами

    Delphi Client/Server Suite 3.0

    Delphi Client/Server Suite 3.0 ориентирован на организации, разрабатывающие корпоративные системы, предназначенные для работы с данными, находящимися в базах данных серверов DB/2, Informix, Interbase, MS SQL Server, Oracle, Sybase. Сочетает в себе высокопроизводительный клиентский инструментарий и широкий набор средств работы с серверами БД. По сравнению с версией Client/Server Suite имеет следующие расширения:

    • Набор компонентов DecisionCube для облегчения анализа данных, имеющих множество размерностей

    • Высокопроизводительные SQL Links с неограниченной лицензией для доступа к данным Oracle, Sybase, Informix, MS SQL Server, InterBase, DB/2

    • BDE Driver Development Kit для открытого доступа к любым механизмам баз данных через BDE

    • Визуальное управление метаданными SQL сервера, включая хранимые процедуры и триггеры, с помощью SQL Database Explorer

    • Тестирование, отладка и настройка производительности приложений, использующих средства SQL с помощью SQL Monitor

    • Автоматическое создание правильных SQL выражений с помощью Visual Query Builder

    • Возможность создавать и тестировать приложения, работающие с InterBase NT (четыре пользовательских лицензии)

    • Легкое деление приложений на отдельные части при помощи Remote DataBroker

    • Управление целостностью данных при помощи ConstraintBroker

    • Высокоскоростное представление данных через WebServer

    • Открытые решения для поддержки Netscape NSAPI и Microsoft ISAPI при помощи WebBridge

    • Централизованная обработка информации при помощи WebModule и обработка запросов при помощи WebDispatch

    • Распространение тонких клиентов, которым не требуется дополнительно ничего (даже BDE), использу WebDeploy

    • Встроенный менеджер версий Intersolve PVCS для групповой разработки

    • Эксперт для интеграции с CASE-средствами

    • Data Pump эксперт для быстрого масштабирования приложений


    2. Какие форматы скомпилированных модулей можно получить в Delphi 3.0?

    Delphi может создавать EXE и DLL для Win32. Естественно, вы может также создавать OCX, но для этого нужно знать соглашения по написанию DLL в формате OCX.

    Delphi может также создавать 32-разрядные консольные приложения для работы под Win32.

    Вы можете создать package — это тоже DLL, но с некоторыми особенностями.

    Если вы создаете ActiveX, то у вас появится файл с расширением DLL, который вы можете при желании переименовать в ActiveX. Также у вас автоматически появится файл TLB — это библиотека типов (Type Library), необходимая для поддержки нестандартных типов данных. Кроме того, вы можете создать TLB файл отдельно.


    3. Какую модель данных использует Delphi?

    Delphi 2.0 использует так называемую плавающую модель памяти (FLOAT), которая принята в Win 32. Отличительной особенностью данной модели памяти является линейная 32-разрядная адресация всего адресного пространства, которое может иметь соответственно размер до 4 Гбайт. При этом все указатели, адреса процедур, указатели на VMT также адресуются через 32-разрядные регистры.

    Object Pascal и Windows API

    1. Каким образом реализована поддержка COM/DCOM?

    В языке Object Pascal появилось понятие интерфейса (interface). Описание интерфейса похоже на описание обычного класса, в нем не может быть указан спецификатор видимости. Список членов ограничен методами и свойствами, для чтения/записи которых используются методы. Поля в интерфейсе недопустимы. Типы параметров и возвращаемых значений ограничены типами, допустимыми в COM/DCOM, их полный перечень можно найти в on-line help.

    Интерфейсы могут наследоваться, в соответствии с идеологией COM, допускается только одиночное наследование. Класс может реализовывать несколько интерфейсов, в этом случае он должен объявить методы, которые по типу параметров и возвращаемых значений совпадают с описанием методов соответствующего интерфейса.

    Все интерфейсы имеют в качестве базового класса интерфейсный тип IUnknown, объявленный в модуле System следующим образом:

    IUnknown = interface

      ['{00000000-0000-0000-C000-000000000046}']

      function QueryInterface(const IID: TGUID; out Obj): Integer; stdcall;

      function _AddRef: Integer; stdcall;

      function _Release: Integer; stdcall;

    end;

    Для поддержки OLE Automation также в модуле System имеется интерфейс IDispatch:

    IDispatch = interface(IUnknown)

      ['{00020400-0000-0000-C000-000000000046}']

      function GetTypeInfoCount(out Count: Integer): Integer; stdcall;

      function GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): Integer; stdcall;

      function GetIDsOfNames(const IID: TGUID; Names: Pointer;

        NameCount, LocaleID: Integer; DispIDs: Pointer): Integer; stdcall;

      function Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer;

        Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): Integer; stdcall;

    end;

    Для облегчения работы с COM имеется класс TInterfacedObject, который реализует методы, объявленные в интерфейсе IUnknown.


    2. В Delphi 3.0 допускается множественное наследование классов?

    Нет, нет и еще 1024 раза нет. Язык Object Pascal в Delphi 3.0 не позволяет вам наследовать класс от более чем одного класса. Строго говоря, вы всегда наследуете ваш новый класс ровно от одного уже имеющегося. Если вы явно не указываете базовый класс, то им становится TObject, стоящий таким образом в иерархии вообще всех классов.

    Слухи о множественном наследовании идут из-за того, что в языке Object Pascal в Delphi 3.0 реализована поддержка интерфейсов для модели COM/DCOM. При этом новый класс может наследоваться от одного класса и реализовывать произвольное количество интефейсов. Соответствующее объявление нового типа:

    TNewObject = class(TBaseObject, ISomeInterface, IAnotherInterface, IDummyInterface)

    вызывает некоторую путаницу и может быть похожа на множественное наследование. Здесь можно провести анологию с языком Java. Интерфейсы в Delphi 3.0 похожи на интерфейсы в Java с отличием в том, что Java допускает множественное наследование интерфейсов, а Delphi — нет.

    Фактически объявление интефейса представляет из себя кусочек таблицы виртуальных методов (VMT), который присоединяется к основной VMT класса.

    Delphi, С++ Builder и базы данных: вопросы и ответы

    Прошу вас рассказать, как подключать к Personal Oracle с помощью BDE. Для меня пока это остается загадкой.

    Доступ к Personаl Oracle (как и к любой другой версии этой СУБД) осуществляется следующим образом.

    Сначала нужно запустить сервер (в случае Personal Oracle для Windows 95 это отдельное приложение, в случае Oracle для Windows NT — набор сервисов, обслуживающих конкретную базу данных) и настроить клиентскую часть Oracle. Для этого следует запустить утилиту SQLNet Easy configuration (в случае Oracle 8 — Oracle Net8 Easy Config) и с ее помощью создать описание псевдонима базы данных Oracle (для него, как и в BDE, используется термин alias, но это не то же самое, что псевдоним BDE). При создании этого описания важны три параметра. Первый из них – сетевой протокол, с помощью которого осуществляется доступ к серверу Oracle (IPX/SPX, TCP/IP и др.). Второй параметр — местоположение сервера в сети. В случае Personal Oracle это обычно компьютер с IP-адресом 127.0.0.1 (это специальный адрес для доступа к локальному компьютеру, так называемый TCP loopback address). Третий параметр — имя базы данных. По умолчанию в случае Personal Oracle она называется ORCL. В общем случае имя может быть любым, но это должно быть имя уже существующей базы данных, с которой вы собираетесь работать. В принципе все описания псевдонимов Oracle хранятся в текстовом файле TNSNAMES.ORA, и можно редактировать его вручную.

    Далее следует запустить утилиту SQL Plus и проверить соединение клиента с сервером. Обычно в качестве имени пользователя используется имя SYSTEM и пароль MANAGER (если вы сами администрируете сервер). Если же сервер был установлен раньше, есть смысл узнать у администратора базы данных, каким именем и паролем следует воспользоваться. Помимо имени пользователя и пароля, SQL Plus запросит так называемую строку связи, в которой должно содержаться имя сервиса, который был создан вами перед этим. При удачном соединении в SQL Plus появится соответствующее сообщение. Отметим, что утилита Oracle Net8 Easy Config позволяет протестировать соединение непосредственно в процессе создания описания сервиса. Если соединение с сервером было неудачным, стоит проверить, поддерживается ли указанный сетевой протокол и виден ли в сети компьютер, на котором установлен сервер, и, если нужно, внести изменения в описание сервиса.

    Далее можно, наконец, заняться настройкой BDE. В качестве Server Name следует указать имя псевдонима Oracle (его можно просто выбрать из выпадающего списка, так как BDE Administrator также обращается к файлу TNSNAMES.ORA). После этого нужно проверить соединение с сервером через BDE с помощью BDE Administrator или SQL Explorer.

    Если соединение не устанавливается и появляется сообщение "Vendor initialization failed", стоит убедиться, что динамическая загружаемая библиотека, указанная в параметре Vendor Init драйвера Oracle, действительно присутствует на данном компьютере. На всякий случай стоит скопировать ее в Windows\System, так как некоторые ранние версии BDE в Windows 95 не находят эту библиотеку в подкаталоге Bin каталога, в котором установлен клиент Oracle, в силу ограничений, налагаемых этой операционной системой на длину переменной окружения PATH. Отметим также, что при использовании Oracle 8 следует использовать версию не ниже 8.0.4; в случае использования более ранней версии следует обновить ее до 8.0.4.

    Недавно я перешел на использование Oracle, и все мои попытки использовать компонент TStoredProc кончаются неудачей. Почему?

    Причины неработоспособности компонента TStoredProc могут быть следующими. Во-первых, при использовании ODBC-доступа может оказаться, что применяемый вами ODBC-драйвер не поддерживает хранимые процедуры (как известно, не все ODBC-драйверы их поддерживают).

    И во-вторых, имеется известная проблема, описание которой содержится в разделе Developers support корпоративного сайта Inprise. Дело в том, что число параметров хранимой процедуры, с которой взаимодействует компонент TStoredProc, не должно превышать 10. В случае, если реальное число параметров превышает 10, многие разработчики переписывают хранимые процедуры так, чтобы они использовали строковые параметры, содержащие по несколько реальных параметров.

    Есть ли возможность в Delphi как-то корректно прервать выполнение SQL запроса к серверу Oracle с помощью BDE? Например, при использованиис SQL Plus после отправки на выполнение SQL-запроса на экране появляется окно с кнопкой Cancel, благодаря чему мы имеем возможность в любой момент прервать выполнение этого запроса. Можно ли что-то подобное сделать в Delphi?

    Насколько мне известно, для этой цели лучше всего использовать функции Oracle Call Interface (низкоуровневый API Oracle). В комплекте поставки Oracle есть соответствующие примеры для C, и переписать их на Pascal несложно.

    Некоторые драйверы SQL Link позволяют прекратить выполнение запроса, если время его выполнения превышает заранее заданное значение (параметр MAX QUERY TIME соответствующего драйвера). Однако драйвер ORACLE, к сожалению, не входит в их число.

    Что необходимо предпринять, чтобы сгенерировать из ERwin таблицы для локальной базы данных Paradox 5.0? На компьютере установлены Delphi 4.0 и MetaBase.

    Для этого требуется установить ODBC-драйвер для этой версии Paradox той же разрядности, что и ERwin. Затем нужно описать соответствующий ODBC-источник, и он будет доступен в ERwin.

    Не могли бы Вы мне подсказать, как заблокировать функцию вставки записи непосредственно в компоненте TDBGrid с сохранением всех остальных возможностей редактирования таблицы.

    Наиболее разумным представляется создать обработчик события BeforeInsert компонента TTable, TQuery или TClientDataSet, данные из которых отображаются в TDBGrid. Сам компонент TDBGrid не имеет подходящего события для обработки, так как это компонент, предназначенный только для создания пользовательского интерфейса, а в данном случае следует, по существу, запретить добавление записей в таблицу.

    У меня в комплект Borland C++ Builder не входит Visual Query Builder. Могу ли я связать две таблицы без него?

    Безусловно, можно связать две таблицы и без VQB. Самый простой способ – запустить Database Form Wizard и связать две таблицы, используя TQuery. Те два запроса, которые при этом получатся (один из них – параметризованный), можно использовать как образец.

    Кроме того, можно просто написать вручную необходимый запрос к любому числу таблиц и поместить его в свойство SQL компонента TQuery. Все инструменты для генерации запросов (Visual Query Builder, SQL Builder и др.) просто предоставляют для этого визуальные средства, а результатом их работы является именно текст запроса, помещаемый в это свойство.

    Я установил Borland C++ Builder 3.0 Client/Server Suite и InterBase Server 5.1.1. (автоматически с ним установился InterBase 5.x Driver by Visigenic). Но у меня не работают хранимые процедуры. То есть процедура правильно откомпилирована, и вызов ее из C++ Builder осуществляется с помощью выполнения оператора

    StoredProc1->ExecProc();

    При этом возникает следующая ошибка : "Capability not supported. General SQL error. [Visigenic][ODBC InterBase 4.x Driver] Driver not capable".

    ODBC-драйвер может не поддерживать хранимые процедуры. В этом случае стоит попытаться использовать драйвер SQL Link (он должен быть в C++ Builder 3.0 Client/Server Suite). Для этого нужно создать для вашей базы данных псевдоним типа INTRBASE. В этом случае хранимые процедуры должны работать.

    Если хранимые процедуры тем не менее остаются недоступными, стоит проверить, что и в какой последовательности было установлено на ваш компьютер. Такие неприятности могут быть, если, например, вы установили какой-либо продукт, написанный на Delphi 2, после C++Builder 3. В этом случае можно переустановить BDE, взяв его последнюю версию на сайте Inprise– все зарегистрированные пользователи C++ Builder 3.0 Client/Server Suite имеют право это сделать.

    При удалении записей из таблицы dBase с помощью компонента TTable они просто приобретают признак удаления, и я никак не могу добиться их физического удаления. Как быть?

    Ваша проблема решается просто — для физического удаления записей нужно использовать функцию DbiPackTable (ее описание есть в справочном файле BDE).

    При добавлении новых записей с помощью TTable.AppendRecord в индексированную таблицу FoxPro через какое-то время (то есть при добавлении сразу большого количества записей) возникает ошибка: "Access to table disabled because of previous error. Read failure. File" <имя_файла.cdx>.

    Возможно, причина заключается в том, что операции чтения-записи в файл, содержащий таблицу FoxPro, особенно при использовании кэширования, предоставляемого операционной системой, конфликтуют с содержимым индексного файла (это часто происходит при многопользовательской работе). Дело в том, что ранние версии dBase, FoxPro, Clipper работали с неподдерживаемыми индексами, то есть индексные файлы не обновлялись одновременно с таблицей, и для их синхронизации требовалось выполнять специальные операции. Но соответствующие средства разработки, применяемые в то время, обычно не поддерживали никаких аналогов транзакций – записи обычно вставлялись по одной.

    В случае применения старых версий формата FoxPro следует избегать кэширования при выполнении дисковых операций с файловым сервером, содержащим базу данных FoxPro. Кроме того, следует проверить и, если необходимо, изменить в настройках BDE параметры MINBUFSIZE, MAXBUFSIZE, LOCAL SHARE – возможно, проблема заключается в недостаточной величине буферов BDE для кэширования данных или в одновременном доступе к данным приложений, использующих и не использующих BDE.

    Еще одним из способов решения этой проблемы (самым радикальным) является замена FoxPro на какую-нибудь из серверных СУБД. Например, IB Database неплохо справляется с одновременным вводом большого количества записей благодаря некоторым специфическим особенностям архитектуры этого сервера.

    Позволяет ли QuickReport выгружать данные в формате Microsoft Excel?

    Quick Report не позволяет выгружать данные в формате Microsoft Excel. Но последние его версии позволяют сохранять отчеты в формате CSV (Comma Separated Value) и HTML, и оба эти формата можно прочесть с помощью Excel.

    Помимо этого, для генерации отчета можно использовать автоматизацию Excel, вообще не прибегая к использованию QuickReport.

    Как можно создать свою форму просмотра отчетов QuickReport в С++Builder?

    Для создания собственных окон просмотра отчета можно использовать компонент TQRPreview. Для этой цели следует создать форму (назовем ее PreviewForm), поместить на нее компонент TQRPreview, сослаться на нее в форме, содержащей отчет, и в форме, из которой вызывается просмотр отчета. Код для показа отчета выгладит примерно так:

    void __fastcall TForm1::Button1Click(TObject *Sender) {

     PreviewForm->Show();

     QuickReport1->Preview();

    }

    Далее создадим обработчик события OnPreview компонента TQuickRep:

    void __fastcall TQuickReport1::QuickReport1Preview(TObject *Sender) {

     PreviewForm->QRPreview1->QRPrinter = QuickReport1->QRPrinter;

    }

    После этого данный отчет будет появляться не в стандартном окне просмотра, а в форме PreviewForm.

    Возможно ли использование компонентов Decision Support System при генерации отчетов в QuickReport и, если можно , то каким образом? Если QuickReport не подходит для этих целей, то какие другие варианты Вы можете посоветовать?

    Самый простой способ – использовать компоненты TQRLabel, текст в которых динамически меняется во время печати (то есть способ, которым можно напечатать все, что угодно, написав при этом немного кода). В принципе можно двумерное сечение куба записать во временную таблицу или в компонент TClientDataSet, написав соответствующий цикл, и сделать отчет на ее основе. Использование DecisionQuery в качестве источника данных для отчета также вполне возможно. Другие возможные варианты – это использование автоматизации Word или Excel, либо вычисление сумм внутри отчета. Можно также использовать другие генераторы отчетов – например, с помощью Crystal Reports можно создавать отчеты, содержащие кросс-таблицы.

    Как корректно подключить Crystal Reports к Delphi?

    В составе Crystal Reports Professional имеется VCL-компонент для Delphi, элемент управления ActiveX, модуль CRPE32.PAS, котором объявлены все функции и структуры Print Engine API, и описание опубликованных методов Crystal Reports как сервера автоматизации. Соответственно есть следующие возможности подключения Crystal Reports к Delphi:

    1. Использование функций Report Engine API из библиотеки CRPE32 DLL. В этом случае следует добавить в проект модуль CRPE32.PAS и сослаться на этот модуль в предложении uses. Ниже приведен пример соответствующего кода:

    procedure TForm1.Button1Click(Sender: TObject);

    VAR RepNam:PChar;

    begin

    if   OpenDialog1.Execute then

    begin

    If PEOpenEngine then

         begin RepNam := StrAlloc(80);

            StrPCopy(RepNam, OpenDialog1.Filename);

            JN := PEOpenPrintJob(RepNam);

              if JN   = 0 then

                   ShowMessage('Ошибка открытия отчета');

                StrDispose(RepNam);

                end

             else

                 ShowMessage('Ошибка открытия отчета');

    end;

    end;

    procedure TForm1.Button2Click(Sender:  TObject);

    begin

    PEClosePrintJob(JN);

    PECloseEngine;

    Close;

    end;


    procedure TForm1.Button3Click(Sender:   TObject);

    begin

    PEOutputToWindow(jn,'Пример использования Crystal Reports Print

    Engine',30,30,600,400,0,0) ;

         if PEStartPrintJob(JN, True) = False then

              ShowMessage('Ошибка   вывода отчета');

    end;

    end.

    Следует помнить, что строковые параметры, передаваемые в функции Print Engine API, представляют собой тип данных PChar, а не стандартные строки, используемые в Pascal, поэтому для передачи таких параметров, как, например, имя отчета, следует осуществить преобразование типов с помощью функции StrPCopy. Для успешной компиляции подобных приложений файл CRPE32.PAS должен находиться в том же каталоге, что и разрабатываемое приложение, либо в каталоге Delphi\Lib.

    2. Использование VCL-компонента и комплекта поставки (для этого следует установить его в палитру компонентов Delphi). Естественно, этот компонент инкапсулирует те же самые функции Print Engine API. Существуют также аналогичные компоненты третьих фирм (например, компонент от SupraSoft Ltd., http://www.suprasoft.com).

    3. Использование элемента управления Crystal Reports ActiveX. Этот элемент управления может быть установлен в палитру компонентов Delphi. Он обладает набором свойств и методов, более или менее сходным с соответствующим VCL-компонентом из комплекта поставки Crystal Reports Professional.

    5. Использование Crystal Reports как сервера автоматизации. В справочной системе Crystal Reports имеется подробное описание иерархии вложенных объектов и их методов (и внушительный набор примеров для Visual Basic, аналоги которых несложно создать и на Pascal). Ниже приведен пример соответствующего кода:

    procedure TForm1.Button1Click(Sender: TObject);

    var r,rep: Variant;

    begin

         rep:=CreateOleObject('Crystal.CRPE.Application');

         r:=rep.OpenReport('d:\Report2.rpt');

         r.RecordSelectionFormula := '{items.ItemNo} = '+Edit1.Text;

         r.Preview;

         r:=Unassigned;

         rep:=Unassigned;

    end;

    6. Можно также сделать отчет в виде исполняемого файла и вызвать его из приложения. Но в этом случае в отчет не удастся передать параметры.

    Delphi VCL FAQ

    Вопрос:

    Как разместить прозрачную надпись на TBitmap?

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     OldBkMode : integer;

    begin

     Image1.Picture.Bitmap.Canvas.Font.Color := clBlue;

     OldBkMode := SetBkMode(Image1.Picture.Bitmap.Canvas.Handle,TRANSPARENT);

     Image1.Picture.Bitmap.Canvas.TextOut(10, 10, 'Hello');

     SetBkMode(Image1.Picture.Bitmap.Canvas.Handle,OldBkMode);

    end;


    Вопрос:

    Можно ли обратиться к колонке или строке grid'а по заголовку?


    Ответ:

    В следующем примере приведены две функции: GetGridColumnByName() и GetGridRowByName(), которые возвращают колонку или строку, имеющую заданный заголовок (caption).

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    begin

     StringGrid1.Rows[1].Strings[0] := 'This Row';

     StringGrid1.Cols[1].Strings[0] := 'This Column';

    end;


    function GetGridColumnByName(Grid : TStringGrid; ColName : string): integer;

    var

     i : integer;

    begin

     for i := 0 to Grid.ColCount - 1 do
    if Grid.Rows[0].Strings[i] = ColName then 
    begin

      Result := i;

      exit;

     end;

     Result := -1;

    end;


    function GetGridRowByName(Grid : TStringGrid; RowName : string): integer;

    var

     i : integer;

    begin

     for i := 0 to Grid.RowCount - 1 do
    if Grid.Cols[0].Strings[i] = RowName then
    begin

      Result := i;

      exit;

     end;

     Result := -1;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    var

     Column : integer;

     Row : integer;

    begin

     Column := GetGridColumnByName(StringGrid1, 'This Column');

     if Column = -1 then
    ShowMessage('Column not found')

     else
    ShowMessage('Column found at ' + IntToStr(Column));

     Row := GetGridRowByName(StringGrid1, 'This Row');

     if Row = -1 then
    ShowMessage('Row not found')

     else
    ShowMessage('Row found at ' + IntToStr(Row));

    end;


    Вопрос:

    Как использовать клавишу-акселератор в TTabsheets? Я добавляю клавишу-акселератор в заголовок каждого Tabsheet моего PageControl, но при попытке переключать страницы этой клавишей программа пикает и ничего не происходит.

    Ответ: Можно перехватить сообщение CM_DIALOGCHAR.

    Пример:

    type TForm1 = class(TForm)

     PageControl1: TPageControl;

     TabSheet1: TTabSheet;

     TabSheet2: TTabSheet;

     TabSheet3: TTabSheet;

    private {Private declarations}

     procedure CMDialogChar(var Msg:TCMDialogChar); message CM_DIALOGCHAR;

    public

      {Public declarations}

    end;


    var

      Form1: TForm1;


    implementation

    {$R *.DFM}

    procedure TForm1.CMDialogChar(var Msg:TCMDialogChar);

    var

      i : integer;

    begin

     with PageControl1 do begin

      if Enabled then for i := 0 to PageControl1.PageCount - 1 do if ((IsAccel(Msg.CharCode, Pages[i].Caption)) and (Pages[i].TabVisible)) then begin

       Msg.Result:=1;

       ActivePage := Pages[i];

       exit;

      end;

     end;

     inherited;

    end;


    Вопрос:

    При использованиии компонента TRegistry под NT пользователь с права доступа ниже чем "администратор" не может получить доступа к информации реестра в ключе HKEY_LOCAL_MACHINE. Как это обойти?

    Ответ:

    Проблема вызвана тем, что TRegistry всегда открывает реестр с параметром KEY_ALL_ACCESS (полный доступ), даже если необходим доступ KEY_READ (только чтение). Избежать этого можно используя функции API для работы с реестром (RegOpenKey и т.п.), или создать новый класс из компонента TRegestry, и изменить его так чтобы можно было задавать режим открытия реестра.


    Вопрос:

    Можно ли изменить число колонок и их ширину в компоненте TFileListBox?

    Ответ:

    В приведенном примере FileListBox приводится к типу TDirectoryListBox — таким образом можно добавиь дополнительные колонки.

    Пример:

    with TDirectoryListBox(FileListBox1) do begin

     Columns := 2;

     SendMessage(Handle, LB_SETCOLUMNWIDTH, Canvas.TextWidth('WWWWWWWW.WWW'),0);

    end;


    Вопрос:

    Как настроить табуляцию в компоненте TMemo?

    Ответ:

    Пошлите в Memo сообщение EM_SETTABSTOPS. Например установим первую позицию табуляции на 20-й пиксел.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    var

     DialogUnitsX : LongInt;

     PixelsX : LongInt;

     i : integer;

     TabArray : array[0..4] of integer;

    begin

     Memo1.WantTabs := true;

     DialogUnitsX := LoWord(GetDialogBaseUnits);

     PixelsX := 20;

     for i := 1 to 5 do begin

      TabArray[i - 1] :=((PixelsX * i ) * 4) div DialogUnitsX;

     end;

     SendMessage(Memo1.Handle, EM_SETTABSTOPS,5,LongInt(@TabArray));

     Memo1.Refresh;

    end;


    Вопрос:

    Как перехватить нажатия функциональных клавиш и стрелок?

    Ответ:

    Проверяйте значение переменной key на равенство VK_RIGHT, VK_LEFT, VK_F1 и т.д. на событии KeyDown формы.

    Пример:

    procedure TForm1.FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);

    begin

     if Key = VK_RIGHT then Form1.Caption := 'Right';

     if Key = VK_F1 then Form1.Caption := 'F1';

    end;


    Вопрос:

    При обработке события DrawCell компонента DrawGrid я пишу Font.Color := clRed; и получаю бесконечный цикл мерцаний. Почему?

    Ответ:

    Правильно укажите границы используемого канваса.

    Пример:

    If (Row = 0) then begin

     DrawGrid1.Canvas.Font.Color := clRed;

     DrawGrid1.Canvas.TextOut(Rect.Left,Rect.Top, IntToStr(Col));

    end;


    Вопрос:

    При использовании BitBtn Caption(текст) и картинка(bitmap) из файла не видны одновременно. Почему?

    Ответ:

    Это может происходить если картинка слишком велика. Класс TBitBtn сначала рисует картинку, а затем выводит текст над, под, слева или справа от картинки (в завивимости от свойства Layout). Если размер картинки такой же как у всей кнопки для вывода текста просто не остается места. Если Вам нужно получить кнопку такого же размера как Ваша картинка и видеть при этом надпись на кнопке Вам придется выводить текст надписи непосредственно на канву картинки.

    Пример:

    var

     bm : TBitmap;

     OldBkMode : integer;

    begin

     bm := TBitmap.Create;

     bm.Width := BitBtn1.Glyph.Width;

     bm.Height := BitBtn1.Glyph.Height;

     bm.Canvas.Draw(0, 0, BitBtn1.Glyph);

     OldBkMode := SetBkMode(bm.Canvas.Handle, Transparent);

     bm.Canvas.TextOut(0, 0, 'The Caption');

     SetBkMode(bm.Canvas.Handle, OldBkMode);

     BitBtn1.Glyph.Assign(bm);

    end;


    Вопрос:

    Можно ли изменить вид текстового курсора (каретки) edit'а или другого элемента управления Windows?

    Ответ:

    Можно! В примере показано как создать два цветных "bitmap'а": "улыбчивый" и "хмурый" и присвоить их курсору edit'а. Для этого нужно перехватить оконную процедуру edit'а. Чтобы сделать это заменим адрес оконной процедуры Edit'а нашим собственным, а старую оконную процедуру будем вызывать по необходимости. Пример показывает "улыбчивый" курсор при наборе текста и "хмурый" при забое клавишей backspace.

    Пример:

    unit caret1;

    interface

    {$IFDEF WIN32}

    uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls;

    {$ELSE}

    uses WinTypes, WinProcs, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls;

    {$ENDIF}

    type TForm1 = class(TForm)

     Edit1: TEdit;

     procedure FormCreate(Sender: TObject);

     procedure FormDestroy(Sender: TObject);

    private

     {Private declarations}

    public

     {Public declarations}

     CaretBm : TBitmap;

     CaretBmBk : TBitmap;

     OldEditsWindowProc : Pointer;

    end;


    var Form1: TForm1;


    implementation

    {$R *.DFM}


    type

    {$IFDEF WIN32}

     WParameter = LongInt;

    {$ELSE}

     WParameter = Word;

    {$ENDIF}

    LParameter = LongInt;


    {New windows procedure for the edit control}

    function NewWindowProc(WindowHandle : hWnd; TheMessage : WParameter; ParamW : WParameter; ParamL : LParameter) : LongInt

    {$IFDEF WIN32} stdcall; {$ELSE} ; export; {$ENDIF}

    begin

     {Call the old edit controls windows procedure}

     NewWindowProc := CallWindowProc(Form1.OldEditsWindowProc, WindowHandle, TheMessage, ParamW, ParamL);

     if TheMessage = WM_SETFOCUS then begin

      CreateCaret(WindowHandle, Form1.CaretBm.Handle, 0, 0);

      ShowCaret(WindowHandle);

     end;

     if TheMessage = WM_KILLFOCUS then begin

      HideCaret(WindowHandle);

      DestroyCaret;

     end;

     if TheMessage = WM_KEYDOWN then begin

      if ParamW = VK_BACK then CreateCaret(WindowHandle, Form1.CaretBmBk.Handle, 0, 0)

      else CreateCaret(WindowHandle, Form1.CaretBm.Handle, 0, 0);

      ShowCaret(WindowHandle);

     end;

    end;


    procedure TForm1.FormCreate(Sender: TObject);

    begin

     {Create a smiling bitmap using the wingdings font}

     CaretBm := TBitmap.Create;

     CaretBm.Canvas.Font.Name := 'WingDings';

     CaretBm.Canvas.Font.Height := Edit1.Font.Height;

     CaretBm.Canvas.Font.Color := clWhite;

     CaretBm.Width := CaretBm.Canvas.TextWidth('J') + 2;

     CaretBm.Height := CaretBm.Canvas.TextHeight('J') + 2;

     CaretBm.Canvas.Brush.Color := clBlue;

     CaretBm.Canvas.FillRect(Rect(0, 0, CaretBm.Width, CaretBm.Height));

     CaretBm.Canvas.TextOut(1, 1, 'J');

     {Create a frowming bitmap using the wingdings font}

     CaretBmBk := TBitmap.Create;

     CaretBmBk.Canvas.Font.Name := 'WingDings';

     CaretBmBk.Canvas.Font.Height := Edit1.Font.Height;

     CaretBmBk.Canvas.Font.Color := clWhite;

     CaretBmBk.Width := CaretBmBk.Canvas.TextWidth('L') + 2;

     CaretBmBk.Height := CaretBmBk.Canvas.TextHeight('L') + 2;

     CaretBmBk.Canvas.Brush.Color := clBlue;

     CaretBmBk.Canvas.FillRect(Rect(0,0, CaretBmBk.Width, CaretBmBk.Height));

     CaretBmBk.Canvas.TextOut(1, 1, 'L');

     {Hook the edit controls window procedure}

     OldEditsWindowProc := Pointer(SetWindowLong(Edit1.Handle,GWL_WNDPROC, LongInt(@NewWindowProc)));

    end;


    procedure TForm1.FormDestroy(Sender: TObject);

    begin

     {Unhook the edit controls window procedure and clean up}

     SetWindowLong(Edit1.Handle,GWL_WNDPROC, LongInt(OldEditsWindowProc));

     CaretBm.Free;

     CaretBmBk.Free;

    end;


    Вопрос:

    При использовании модулей доступа к BDE (DbiTypes, DbiProcs, DbiErrs), любая попытка вызвать процедуру abort выдает ошибку при компиляции при вызове метода abort "Statement expected, but expression of type 'Integer' found". Я пытался найти DbiTypes.pas, DbiProcs.pas и DbiErrs.pas чтобы разобраться но не нашел этих файлов. Где расположены эти файлы и как обойти ошибку?

    Ответ:

    Модули DbiTypes, DbiProcs, DbiErrs это псевдонимы модуля "BDE", обьявлены в Projects→Options→Directories/Conditionals→Unit Aliases. Исходник модуля DBE находится в каталоге "doc" и называется "BDE.INT". В этом файле обьявленна константа ABORT со значением -2. Так как Вы хотите использовать процедуру Abort(), которая обьявлена в модуле SysUtils, Вам нужно добавить префикс SysUtils перед вызовом процедуры Abort.

    Пример:

    SysUtils.Abort;


    Вопрос:

    Почему при изменении цвета букв StatusBar'а ничего не происходит?

    Ответ:

    Status bar — стандартный элемент управления Windows, и соответственно цвет его букв — значение clBtnText которое изменяется с помощью настроек в Control Panel. Этот цвет черный по умолчанию и может изменяться в зависимости от выбранной цветовой схемы. Другие стандартные элемент управления Windows, например кнопки, также имеют цвет букв, настраиваемый из ControlPanel. StatusBar и его панели имеют свойство "owner-draw", позволяющее Вам использовать любой цвет букв.

    Пример:

    procedure TForm1.StatusBar1DrawPanel(StatusBar: TStatusBar; Panel: TStatusPanel; const Rect: TRect);

    begin

     if Panel = StatusBar.Panels[0] then begin

      StatusBar.Canvas.Font.Color := clRed;

      StatusBar.Canvas.TextOut(Rect.Left, Rect.Top, 'Panel - 0')

     end else begin

      StatusBar.Canvas.Font.Color := clGreen;

      StatusBar.Canvas.TextOut(Rect.Left, Rect.Top, 'Panel - 1');

     end;

    end;


    Вопрос:

    Как сделать многострочную надпись на TBitBtn?

    Ответ:

    Выводите текст надписи непосредственно на "glyph" TBitBtn'а. См. пример.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    var

     R : TRect;

     N : Integer;

     Buff : array[0..255] of Char;

    begin

     with BitBtn1 do begin

      Caption := 'A really really long caption';

      Glyph.Canvas.Font := Self.Font;

      Glyph.Width := Width - 6;

      Glyph.Height := Height - 6;

      R := Bounds(0, 0, Glyph.Width, 0);

      StrPCopy(Buff, Caption);

      Caption := '';

      DrawText(Glyph.Canvas.Handle,Buff,StrLen(Buff),R, DT_CENTER or DT_WORDBREAK or DT_CALCRECT);

      OffsetRect(R,(Glyph.Width - R.Right) div 2, (Glyph.Height - R.Bottom) div 2);

      DrawText(Glyph.Canvas.Handle,Buff,StrLen(Buff),R, DT_CENTER or DT_WORDBREAK);

     end;

    end;


    Вопрос:

    Как изменить стиль шрифта RichEdit нажатиями соответствующих комбинаций клавиш? (например включить курсив по нажатию Ctrl + I)

    Ответ:

    В примере стили шрифта меняются по нажатию след. комбинаций клавиш

    • Ctrl + B — вкл/выкл жирного шрифта

    • Ctrl + I — вкл/выкл наклонного шрифта

    • Ctrl + S — вкл/выкл зачеркнутого шрифта

    • Ctrl + U — вкл/выкл подчеркнутого шрифта

    Пример:

    const

     KEY_CTRL_B = 02;

     KEY_CTRL_I = 9;

     KEY_CTRL_S = 19;

     KEY_CTRL_U = 21;


    procedure TForm1.RichEdit1KeyPress(Sender: TObject; var Key: Char);

    begin

     case Ord(Key) of

     KEY_CTRL_B: begin

      Key := #0;

      if fsBold in (Sender as TRichEdit).SelAttributes.Style then (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style - [fsBold]

      else (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style + [fsBold];

     end;

     KEY_CTRL_I: begin

      Key := #0;

      if fsItalic in (Sender as TRichEdit).SelAttributes.Style then (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style - [fsItalic]

      else (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style + [fsItalic];

     end;

     KEY_CTRL_S: begin

      Key := #0;

      if fsStrikeout in (Sender as TRichEdit).SelAttributes.Style then (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style-[fsStrikeout]

      else (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style+[fsStrikeout];

     end;

     KEY_CTRL_U: begin

      Key := #0;

      if fsUnderline in (Sender as TRichEdit).SelAttributes.Style then (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style-[fsUnderline]

      else (Sender as TRichEdit).SelAttributes.Style := (Sender as TRichEdit).SelAttributes.Style+[fsUnderline];

     end;

     end;

    end; 


    Вопрос:

    В документации компонента TRegIniFile говорится, что можно изменять корневой ключ (root key). Я пытаюсь это сделать но ничего не получается.

    Ответ:

    См. пример.

    Пример:

    uses Registry;


    procedure TForm1.Button1Click(Sender: TObject);

    var WinIni : TRegIniFile;

    begin

     WinIni := TRegIniFile.Create('');

     WinIni.RootKey := HKEY_LOCAL_MACHINE;

     WinIni.WriteString('Frank','Borland','Writes Fast Code!');

     WinIni.Free;

    end;


    Вопрос:

    Можно ли динамически изменять свойство "owner" компонента во время выполнения программы?

    Ответ:

    Вы можете менять свойство "owner" и после создания компонента с помощью методов InsertComponent() и RemoveComponent().


    Вопрос:

    Как очистить содержимое Canvas'а?

    Ответ:

    Просто нарисуйте прямоугольник любого цвета.

    Пример:

    Canvas.Brush.Color := ClWhite;

    Canvas.FillRect(Canvas.ClipRect); 


    Вопрос:

    Можно ли динамически менять какая форма считается главной в приложении во время работы программы?

    Ответ:

    Можно, но только во время загрузки приложения. Чтобы сделать это выберите "View→Project Source" и измените код инициализации приложения, так что порядок создания форм зависил от какого-то условия.

    Примечание: Вам придется редактировать этот код, если Вы добавите в приложение новые формы.

    begin

     Application.Initialize;

     if <какое-то условие> then begin

      Application.CreateForm(TForm1, Form1);

      Application.CreateForm(TForm2, Form2);

     end else begin

      Application.CreateForm(TForm2, Form2);

      Application.CreateForm(TForm1, Form1);

     end;

    end.

    Application.Run;


    Вопрос:

    Как программно "щелкнуть" по компоненту speed button? Я пытался использовать SendMessage но у Speedbuttons нет "handle".

    Ответ:

    В примере используется метод Perform класса TControl для отправки сообщения.

    Пример:

    procedure TForm1.SpeedButton1Click(Sender: TObject);

    begin

     ShowMessage('clicked');

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     SpeedButton1.Perform(WM_LBUTTONDOWN, 0, 0);

     SpeedButton1.Perform(WM_LBUTTONUP, 0, 0);

    end; 


    Вопрос:

    Можно ли отключить определенный элемент в RadioGroup?

    Ответ:

    В примере показано как получить доступ к отдельным элементам компонента TRadioGroup.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     TRadioButton(RadioGroup1.Controls[1]). Enabled := False;

    end; 


    Вопрос:

    Почему методы рисования Delphi (например MoveTo и LineTo) рисуют на один пиксел короче?

    Ответ:

    Так работает большинство графических систем, включая Windows. Библиотека VCL просто передает вызовы в функции GDI. Если Вы хотите нарисовать линию с последним пикселом включительно просто добавте единицу к координатам.


    Вопрос:

    Как показать подсказки "hints" для элементов меню?

    Ответ:

    В примере создается обработчик события Application.Hint — подсказки меню изображаются на status panel.

    Пример:

    type TForm1 = class(TForm)

     Panel1: TPanel;

     MainMenu1: TMainMenu;

     MenuItemFile: TMenuItem;

     MenuItemOpen: TMenuItem;

     MenuItemClose: TMenuItem;

     OpenDialog1: TOpenDialog;

     procedure FormCreate(Sender: TObject);

     procedure MenuItemCloseClick(Sender: TObject);

     procedure MenuItemOpenClick(Sender: TObject);

    private

     {Private declarations}

     procedure HintHandler(Sender: TObject);

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.FormCreate(Sender: TObject);

    begin

     Panel1.Align := alBottom;

     MenuItemFile.Hint := 'File Menu';

     MenuItemOpen.Hint := 'Opens A File';

     MenuItemClose.Hint := 'Closes the Application';

     Application.OnHint := HintHandler;

    end;


    procedure TForm1.HintHandler(Sender: TObject);

    begin

     Panel1.Caption := Application.Hint;

    end;


    procedure TForm1.MenuItemCloseClick(Sender: TObject);

    begin

     Application.Terminate;

    end;


    procedure TForm1.MenuItemOpenClick(Sender: TObject);

    begin

     if OpenDialog1.Execute then Form1.Caption := OpenDialog1.FileName;

    end;


    Вопрос:

    Как опеделить состояние списка ComboBox, выпал/скрыт?

    Ответ:

    Пошлите ComboBox сообщение CB_GETDROPPEDSTATE.

    Пример:

    if SendMessage(ComboBox1.Handle, CB_GETDROPPEDSTATE,0,0) = 1 then begin

     {список ComboBox выпал}

    end;


    Вопрос:

    Как удалить каталог вместе со всеми содержащимися в нем файлами?

    Ответ:

    В примере стираются все файлы в каталоге и сам каталог. Чтобы удалить файл, помечанные только для чтения (read only) и занятые другими программами в момент удаления — напишите дополнительную процедуру.

    procedure TForm1.Button1Click(Sender: TObject);

    var

     DirInfo: TSearchRec;

     r: integer;

    begin

     r := FindFirst('C:\Download\*.*', FaAnyfile, DirInfo);

     while r = 0 do begin

      if ((DirInfo.Attr and FaDirectory <> FaDirectory) and (DirInfo.Attr and FaVolumeId <> FaVolumeID)) then if DeleteFile(pChar('C:\Download\' + DirInfo.Name))= false then ShowMessage('Unable to delete: C:\Download\'+DirInfo.Name);

      r := FindNext(DirInfo);

     end;

     SysUtils.FindClose(DirInfo);

     if RemoveDirectory('C:\Download\') = false then ShowMessage('Unable to delete directory: C:\Download\');

    end;


    Вопрос:

    Как отключить системное меню формы и кнопки Minimize, Maximize, and Close во время выполнения(Runtime)?

    Ответ:

    В приведенном примере показано как это сделать

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     {Disable}

     Form1.BorderIcons := Form1.BorderIcons - [biSystemMenu, biMinimize, biMaximize];

    end;


    procedure TForm1.Button2Click(Sender: TObject);

    begin

     {Enable}

     Form1.BorderIcons := Form1.BorderIcons + [biSystemMenu, biMinimize, biMaximize];

    end; 


    Вопрос:

    Как извлечь Red, Green, и Blue компонент из определенного цвета?

    Ответ:

    Используйте функции Window API Get RValue(), GetGValue(), и GetBValue().

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Form1.Canvas.Pen.Color := clRed;

     Memo1.Lines.Add('Red := ' + IntToStr(GetRValue(Form1.Canvas.Pen.Color)));

     Memo1.Lines.Add('Red := ' + IntToStr(GetGValue(Form1.Canvas.Pen.Color)));

     Memo1.Lines.Add('Blue:= ' + IntToStr(GetBValue(Form1.Canvas.Pen.Color)));

    end;


    Вопрос:

    Как определить номер текущей строки в TMemo?

    Ответ:

    Чтобы определить номер текущей строки любого объекта управления edit — пошлите ей сообщение EM_LINEFROMCHAR

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var LineNumber : integer;

    begin

     LineNumber := SendMessage(Memo1.Handle, EM_LINEFROMCHAR, word(-1), 0);

     ShowMessage(IntToStr(LineNumber));

    end;


    Вопрос:

    Как проигрывать MPEG файл в Delphi-программе?

    Ответ:

    Если в системе Windows MMSystem установлен декодер MPEG — используя компонент TMediaPlayer

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     MediaPlayer1.Filename := 'C:\DownLoad\rsgrow.mpg';

     MediaPlayer1.Open;

     MediaPlayer1.Display := Panel1;

     MediaPlayer1.DisplayRect := Panel1.ClientRect;

     MediaPlayer1.Play;

    end;


    Вопрос:

    Как использовать анимированный курсор?

    Ответ:

    Во первых необходимо получить handle курсора, а затем определить его в массиве курсоров компонента TScreen. Индексы предопределенных курсоров системы отрицательны, пользователь может определить курсор, индекс которого положителен.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var h : THandle;

    begin

     h := LoadImage(0, 'C:\TheWall\Magic.ani', IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE or LR_LOADFROMFILE);

     if h = 0 then ShowMessage('Cursor not loaded')

     else begin

      Screen.Cursors[1] := h;

      Form1.Cursor := 1;

     end;

    end;


    Вопрос:

    Как узнать о нажатии "non-menu" клавиши в момент когда меню показано?

    Ответ:

    Создайте обработчик сообщения WM_MENUCHAR.

    Пример:

    unit Unit1;

    interface

    uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, Menus;


    type TForm1 = class(TForm)

     MainMenu1: TMainMenu;

     One1: TMenuItem;

     Two1: TMenuItem;

     THree1: TMenuItem;

    private

     {Private declarations}

     procedure WmMenuChar(var m : TMessage); message WM_MENUCHAR;

    public

     {Public declarations}

    end;

    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.WmMenuChar(var m : TMessage);

    begin

     Form1.Caption := 'Non standard menu key pressed';

     m.Result := 1;

    end;


    end.


    Вопрос:

    Как определить наличие сопроцессора?

    Ответ:

    В отличие от общепринятого мнения не всее клоны 486/586/686/ и Pentium имеют сопроцессор для вычислений с плавающей запятой. В примере определяется наличие сопроцессора и под Win16 и под Win32.

    Пример:

    {$IFDEF WIN32}

    uses Registry;

    {$ENDIF}


    function HasCoProcesser : bool;

    {$IFDEF WIN32}

    var TheKey : hKey;

    {$ENDIF}

    begin

     Result := true;

    {$IFNDEF WIN32}

     if GetWinFlags and Wf_80x87 = 0 then Result := false;

    {$ELSE}

     if RegOpenKeyEx(HKEY_LOCAL_MACHINE, 'HARDWARE\DESCRIPTION\System\FloatingPointProcessor',0, KEY_EXECUTE, TheKey) <> ERROR_SUCCESS then result := false;

     RegCloseKey(TheKey);

    {$ENDIF}

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     if HasCoProcesser then ShowMessage('Has CoProcessor')

     else ShowMessage('No CoProcessor - Windows Emulation Mode');

    end;


    Вопрос:

    Как узнать серийный номер аудио CD?

    Ответ:

    CD может иметь или не иметь серийный номер и/или универсальный код продукта (Universal Product Code). MCI-расширение Windows предоставляет эту информации с помощью комманды MCI_INFO_MEDIA_IDENTITY command. Эта команда возвращает уникальную ID-строку.

    Пример:

    uses MMSystem, MPlayer;


    procedure TForm1.Button1Click(Sender: TObject);

    var

     mp : TMediaPlayer;

     msp : TMCI_INFO_PARMS;

     MediaString : array[0..255] of char;

     ret : longint;

    begin

     mp := TMediaPlayer.Create(nil);

     mp.Visible := false;

     mp.Parent := Application.MainForm;

     mp.Shareable := true;

     mp.DeviceType := dtCDAudio;

     mp.FileName := 'D:';

     mp.Open;

     Application.ProcessMessages;

     FillChar(MediaString, sizeof(MediaString), #0);

     FillChar(msp, sizeof(msp), #0);

     msp.lpstrReturn := @MediaString;

     msp.dwRetSize := 255;

     ret := mciSendCommand(Mp.DeviceId, MCI_INFO, MCI_INFO_MEDIA_IDENTITY, longint(@msp));

     if Ret <> 0 then begin

      MciGetErrorString(ret, @MediaString, sizeof(MediaString));

      Memo1.Lines.Add(StrPas(MediaString));

     end else Memo1.Lines.Add(StrPas(MediaString));

     mp.Close;

     Application.ProcessMessages;

     mp.free;

    end;


    end.


    Вопрос:

    Как вывести на элемент управления (Window control) текст, содержащий амперсанд — &?

    Ответ:

    Используя два амперсанда подряд. Windows интерпритирует одиночный амперсанд как указание на то, что следующий символ — горячая клавиша (и поддчеркивает следующий символ вместо изображения аперсанда).

    Пример:

    Button1.Caption := 'Черное && Белое';


    Вопрос:

    Как поместить bitmap в Metafile?

    Ответ:

    см. пример

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     m : TmetaFile;

     mc : TmetaFileCanvas;

     b : tbitmap;

    begin

     m := TMetaFile.Create;

     b := TBitmap.create;

     b.LoadFromFile('C:\SomePath\SomeBitmap.BMP');

     m.Height := b.Height;

     m.Width := b.Width;

     mc := TMetafileCanvas.Create(m, 0);

     mc.Draw(0, 0, b);

     mc.Free;

     b.Free;

     m.SaveToFile('C:\SomePath\Test.emf');

     m.Free;

     Image1.Picture.LoadFromFile('C:\SomePath\Test.emf');

    end;


    Вопрос:

    Как узнать, что курсор мыши над моей формой?

    Ответ:

    Можно использовать функцию GetCapture() из Windows API.

    Примечание: Cм. документацию Windows для информации об ограничениях функции GetCapture.

    Пример:

    procedure TForm1.FormDeactivate(Sender: TObject);

    begin

     ReleaseCapture;

    end;


    procedure TForm1.FormMouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

    begin

     If GetCapture = 0 then SetCapture(Form1.Handle);

     if PtInRect(Rect(Form1.Left,Form1.Top,Form1.Left + Form1.Width, Form1.Top + Form1.Height), ClientToScreen(Point(x, y))) then Form1.Caption := 'Мышка над формой!'

     else Form1.Caption := 'Мышка вне формы...';

    end;


    Вопрос:

    Как программно определить, что приложение работает под Windows NT?

    Ответ:

    см. пример

    Пример:

    function IsNT : bool;

    var osv : TOSVERSIONINFO;

    begin

     result := true;

     GetVersionEx(osv);

     if osv.dwPlatformId = VER_PLATFORM_WIN32_NT then exit;

     result := false;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     if IsNt then ShowMessage('Running on NT')

     else ShowMessage('Not Running on NT');

    end;


    Вопрос:

    Как создать bitmap из пиктогрммы (icon)?

    Ответ:

    Используя Bitmap.Canvas.Draw нарисуйте пиктограмму на Bitmap'е.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     TheIcon : TIcon;

     TheBitmap : TBitmap;

    begin

     TheIcon := TIcon.Create;

     TheIcon.LoadFromFile('C:\Program Files\Borland\IcoCur32\EARTH.ICO');

     TheBitmap := TBitmap.Create;

     TheBitmap.Height := TheIcon.Height;

     TheBitmap.Width := TheIcon.Width;

     TheBitmap.Canvas.Draw(0, 0, TheIcon);

     Form1.Canvas.Draw(10, 10, TheBitmap);

     TheBitmap.Free; TheIcon.Free;

    end;


    Вопрос:

    Как создать отдельную подсказку (hint) для каждой ячейки StringGrid?

    Ответ:

    В приведенном примере отслеживается движение курсора мыши — при перемещении между ячейками StringGrid'а — появляется окно подсказки (hint), показывающее номер текущей строки и колонки.

    Пример:

    type TForm1 = class(TForm)

     StringGrid1: TStringGrid;

     procedure StringGrid1MouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

     procedure FormCreate(Sender: TObject);

    private

     {Private declarations}

     Col : integer;

     Row : integer;

    public

     {Public declarations}

    end;

    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.FormCreate(Sender: TObject);

    begin

     StringGrid1.Hint := '0 0';

     StringGrid1.ShowHint := True;

    end;


    procedure TForm1.StringGrid1MouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

    var

     r : integer;

     c : integer;

    begin

     StringGrid1.MouseToCell(X, Y, C, R);

     with StringGrid1 do begin

      if ((Row <> r) or (Col <> c)) then begin

       Row := r;

       Col := c;

       Application.CancelHint;

       StringGrid1.Hint :=IntToStr(r)+#32+IntToStr(c);

      end;

     end;

    end;


    Вопрос:

    Как внести изменения в код VCL?

    Ответ:

    Примечание: внесение изменений в VCL не поддерживается Borland или Borland Developer Support.

    Но если Вы решили сделать это...

    Изменения в код VCL никогда не должны вносится в секцию "interface" модуля — только в секцию "implimentation". Наиболее безопасный способ внести изменения в VCL — создать новый каталог названный "исправленный VCL". Скопируйте файл VCL который Вы хотите изменить в этот каталог. Внесите изменения (лучше прокомментировать их) в этот файл. Затем добавьте путь к Вашему каталогу "исправленный VCL" в самое начало "library path". Перезапустите Delphi/C++ Builder и перекомпилируйте Ваш проект. "library path" можно изменить в меню:

    Delphi 1 : Options | Environment | Library

    Delphi 2 : Tools | Options | Library

    Delphi 3 : Tools | Environment Options | Library

    Delphi 4 : Tools | Environment Options | Library C++

    Builder : Options | Environment | Library 


    Вопрос:

    Как в Delphi реализовать функцию — эквивалент TwipsPerPixel из VisualBasic?

    Ответ:

    Функции TwipsPerPixelX и TwipsPerPixelY, приведенные в примере реализуют ту же функциональность в Delphi.

    Пример:

    function TwipsPerPixelX(Canvas : TCanvas) : Extended;

    begin

     result := 1440 / GetDeviceCaps(Canvas.Handle, LOGPIXELSX);

    end;


    function TwipsPerPixelY(Canvas : TCanvas) : Extended;

    begin

     result := 1440 / GetDeviceCaps(Canvas.Handle, LOGPIXELSY);

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     ShowMessage(FloatToStr(TwipsPerPixelX(Form1.Canvas)));

     ShowMessage(FloatToStr(TwipsPerPixelY(Form1.Canvas)));

    end;


    Вопрос:

    Как вставить содержимое файла в текущую позицию курсора в компонете TMemo?

    Ответ:

    Считайте файл в TMemoryStream, затем используйте метод TMemo SetSelTextBuf() для вставки текста;

    var

     TheMStream : TMemoryStream;

     Zero : char;

    begin

     TheMStream := TMemoryStream.Create;

     TheMStream.LoadFromFile('C:\AUTOEXEC.BAT');

     TheMStream.Seek(0, soFromEnd); //Null terminate the buffer!

     Zero := #0;

     TheMStream.Write(Zero, 1);

     TheMStream.Seek(0, soFromBeginning);

     Memo1.SetSelTextBuf(TheMStream.Memory);

     TheMStream.Free;

    end;


    Вопрос:

    Как в компоненте TMemo перехватить нажатие Ctrl-V и вставить специальный текст не из буфера обмена (clipboard)?

    Ответ:

    См. пример.

    Пример:

    uses ClipBrd;


    procedure TForm1.Memo1KeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);

    begin

     if ((Key = ord('V')) and (ssCtrl in Shift)) then begin

      if Clipboard.HasFormat(CF_TEXT) then ClipBoard.Clear;

      Memo1.SelText := 'Delphi is RAD!';

      key := 0;

     end;

    end;


    Вопрос:

    Как создать эквивалент TEdit но только с выравниваением вводимого текста по центру или по правой стороне?

    Ответ:

    TEdit не поддерживает выравниваение текста по центру и по правой стороне — лучше использовать компонент TMemo. Вам понадобится запретить пользователю нажимать Enter, Ctrl-Enter и всевозможные комбинации клавиш со стрелками, чтобы избежать появления нескольких строк в Memo. Этого можно добиться и просматривая содержимое текста в TMemo в поисках кода возврата каретки (13) и перевода строки(10) на событиях TMemo Change и KeyPress. Можно также заменять код возврата каретки на пробел — для того чтобы позволять вставку из буфера обмена многострочного текста в виде одной строки.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    begin

     Memo1.Alignment := taRightJustify;

     Memo1.MaxLength := 24;

     Memo1.WantReturns := false;

     Memo1.WordWrap := false;

    end;


    procedure MultiLineMemoToSingleLine(Memo : TMemo);

    var t : string;

    begin

     t := Memo.Text;

     if Pos(#13, t) > 0 then begin

      while Pos(#13, t) > 0 do delete(t, Pos(#13, t), 1);

      while Pos(#10, t) > 0 do delete(t, Pos(#10, t), 1);

      Memo.Text := t;

     end;

    end;


    procedure TForm1.Memo1Change(Sender: TObject);

    begin

     MultiLineMemoToSingleLine(Memo1);

    end;


    procedure TForm1.Memo1KeyPress(Sender: TObject; var Key: Char);

    begin

     MultiLineMemoToSingleLine(Memo1);

    end;


    Вопрос:

    Как запрограммировать undo?

    Ответ:

    См. пример

    Memo1.Perform(EM_UNDO, 0, 0);

    Если Вы хотите узнать, возможно ли выполнить операцию "Undo", проверьте "Undo status":

    If Memo1.Perform(EM_CANUNDO, 0, 0) <> 0 then begin

     {Undo is possible}

    end;

    Для выполнения "Redo" выполните "Undo" еще раз.


    Вопрос:

    Можно ли создать форму, которая получает дополнительные параметры в методе Сreate?

    Ответ:

    Просто замените конструктор Create класса Вашей формы.

    Пример:

    unit Unit2;

    interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs;


    type TForm2 = class(TForm)

    private

     {Private declarations}

    public

     constructor CreateWithCaption(aOwner: TComponent; aCaption: string);

     {Public declarations}

    end;

    var Form2: TForm2;


    implementation

    {$R *.DFM}


    constructor TForm2.CreateWithCaption(aOwner: TComponent; aCaption: string);

    begin

     Create(aOwner);

     Caption := aCaption;

    end;


    uses Unit2;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Unit2.Form2 :=Unit2.TForm2.CreateWithCaption(Application, 'My Caption');

     Unit2.Form2.Show;

    end;


    Вопрос:

    Как бы мне создать эдакий trackbar в котором вместо широкой белой полоски с ползунком была бы тонкая линия?

    Ответ:

    В примере создается компонент, унаследованный от TTrackbar который переопределяет метод CreateParams и убират флаг TBS_ENABLESELRANGE из Style. Константа TBS_ENABLESELRANGE обьявленна в модуле CommCtrl.

    Пример:

    uses CommCtrl, ComCtrls;

    type TMyTrackBar = class(TTrackBar)

     procedure CreateParams(var Params: TCreateParams); override;

    end;


    procedure TMyTrackBar.CreateParams(var Params: TCreateParams);

    begin

     inherited;

     Params.Style := Params.Style and not TBS_ENABLESELRANGE;

    end;


    var MyTrackbar : TMyTrackbar;

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     MyTrackBar := TMyTrackbar.Create(Form1);

     MyTrackbar.Parent := Form1;

     MyTrackbar.Left := 100;

     MyTrackbar.Top := 100;

     MyTrackbar.Width := 150;

     MyTrackbar.Height := 45;

     MyTrackBar.Visible := true;

    end;


    Вопрос:

    Мне нужен временный canvas, но когда я пытаюсь его создать получаю сообщения об ошибках. Как создать TCanvas?

    Ответ:

    Создайте Bitmap и используйте свойство canvas класса TBitmap. Пример создает Bitmap, рисует на его canvas'е, выводит этот canvas на форму и освобождает bitmap.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var bm : TBitmap;

    begin

     bm := TBitmap.Create;

     bm.Width := 100;

     bm.Height := 100;

     bm.Canvas.Brush.Color := clRed;

     bm.Canvas.FillRect(Rect(0, 0, 100, 100));

     bm.Canvas.MoveTo(0, 0);

     bm.Canvas.LineTo(100, 100);

     Form1.Canvas.StretchDraw(Form1.ClientRect,Bm);

     bm.Free;

    end;


    Вопрос:

    В некоторых видео режимах прозрачная часть glyph'а стандартного TBitBtn становится видной. Как этого избежать?

    Ответ:

    В примере используется техника закраски прозрачной части glyph'а цветом кнопки на которой он находится — таким образом glyph кажется прозрачным.

    Пример:

    function InitStdBitBtn(BitBtn : TBitBtn; kind : TBitBtnKind) : bool;

    var

     Bm1 : TBitmap;

     Bm2 : TBitmap;

    begin

     Result := false;

     if Kind = bkCustom then exit;

     Bm1 := TBitmap.Create;

     case Kind of

     bkOK : Bm1.Handle := LoadBitmap(hInstance, 'BBOK');

     bkCancel : Bm1.Handle := LoadBitmap(hInstance, 'BBCANCEL');

     bkHelp : Bm1.Handle := LoadBitmap(hInstance, 'BBHELP');

     bkYes : Bm1.Handle := LoadBitmap(hInstance, 'BBYES');

     bkNo : Bm1.Handle := LoadBitmap(hInstance, 'BBNO');

     bkClose : Bm1.Handle := LoadBitmap(hInstance, 'BBCLOSE');

     bkAbort : Bm1.Handle := LoadBitmap(hInstance, 'BBABORT');

     bkRetry : Bm1.Handle := LoadBitmap(hInstance, 'BBRETRY');

     bkIgnore : Bm1.Handle := LoadBitmap(hInstance, 'BBIGNORE');

     bkAll : Bm1.Handle := LoadBitmap(hInstance, 'BBALL');

     end;

     Bm2 := TBitmap.Create;

     Bm2.Width := Bm1.Width;

     Bm2.Height := Bm1.Height;

     Bm2.Canvas.Brush.Color := ClBtnFace;

     Bm2.Canvas.BrushCopy(Rect(0, 0, bm2.Width, bm2.Height), Bm1, Rect(0, 0, Bm1.width, Bm1.Height), Bm1.canvas.pixels[0,0]);

     Bm1.Free;

     LockWindowUpdate(BitBtn.Parent.Handle);

     BitBtn.Kind := kind;

     BitBtn.Glyph.Assign(bm2);

     LockWindowUpdate(0);

     Bm2.Free;

     Result := true;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     InitStdBitBtn(BitBtn1, bkOk);

    end;


    Вопрос:

    Создание PolyPolygon используя массив точек?

    Ответ:

    Polygon — метод компонента TCanvas получает в качестве параметра динамический массив точек. Функция PolyPolygon() из Windows GDI получает указатель на массив точек.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     ptArray : array[0..9] of TPOINT;

     PtCounts : array[0..1] of integer;

    begin

     PtArray[0] := Point(0, 0);

     PtArray[1] := Point(0, 100);

     PtArray[2] := Point(100, 100);

     PtArray[3] := Point(100, 0);

     PtArray[4] := Point(0, 0);

     PtCounts[0] := 5;

     PtArray[5] := Point(25, 25);

     PtArray[6] := Point(25, 75);

     PtArray[7] := Point(75, 75);

     PtArray[8] := Point(75, 25);

     PtArray[9] := Point(25, 25);

     PtCounts[1] := 5;

     PolyPolygon(Form1.Canvas.Handle, PtArray,PtCounts,2);

    end;


    Вопрос:

    Как создать невизуальный компонент без иконки, которая изображается в палитре компонентов в "design-time" (вроде TField)?

    Ответ:

    Невизуальные компоненты без иконки удобны для субкомпонентов, связанных с какими-то другими компонентами. Создайте компонент как обычно, но используйте RegisterNoIcon вместо RegisterComponent.


    Вопрос:

    Как показывать нестандартный встроенный редактор (inplace editor) в ячейке stringgrid (например combobox).

    Ответ:

    См. пример

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    begin

     {Высоту combobox'а не изменишь, так что вместо combobox'а будем изменять высоту строки grid'а !}

     StringGrid1.DefaultRowHeight := ComboBox1.Height;

     {Спрятать combobox}

     ComboBox1.Visible := False;

     ComboBox1.Items.Add('Delphi Kingdom');

     ComboBox1.Items.Add('Королевство Дельфи');

    end;


    procedure TForm1.ComboBox1Change(Sender: TObject);

    begin

     {Перебросим выбранное в значение из ComboBox в grid}

     StringGrid1.Cells[StringGrid1.Col, StringGrid1.Row] :=ComboBox1.Items[ComboBox1.ItemIndex];

     ComboBox1.Visible := False;

     StringGrid1.SetFocus;

    end;


    procedure TForm1.ComboBox1Exit(Sender: TObject);

    begin

     {Перебросим выбранное в значение из ComboBox в grid}

     StringGrid1.Cells[StringGrid1.Col, StringGrid1.Row] :=ComboBox1.Items[ComboBox1.ItemIndex];

     ComboBox1.Visible := False;

     StringGrid1.SetFocus;

    end;


    procedure TForm1.StringGrid1SelectCell(Sender: TObject; ACol, ARow: Integer; var CanSelect: Boolean);

    var R: TRect;

    begin

     if ((ACol = 3) AND (ARow <> 0)) then begin

      {Ширина и положение ComboBox должно соответствовать ячейке StringGrid}

      R := StringGrid1.CellRect(ACol, ARow);

      R.Left := R.Left + StringGrid1.Left;

      R.Right := R.Right + StringGrid1.Left;

      R.Top := R.Top + StringGrid1.Top;

      R.Bottom := R.Bottom + StringGrid1.Top;

      ComboBox1.Left := R.Left + 1;

      ComboBox1.Top := R.Top + 1;

      ComboBox1.Width := (R.Right + 1) - R.Left;

      ComboBox1.Height := (R.Bottom + 1) - R.Top;

      {Покажем combobox}

      ComboBox1.Visible := True;

      ComboBox1.SetFocus;

     end;

     CanSelect := True;

    end;


    Вопрос:

    Как узнать есть ли в заданном CD-ROM'е Audio CD?

    Ответ:

    Можно использовать функцию Windows API GetDriveType() чтобы определить является ли дисковод CD-ROM'мом. И функцию API GetVolumeInformation() чтобы проверить VolumeName на равенство 'Audio CD'.

    Пример:

    function IsAudioCD(Drive : char) : bool;

    var

     DrivePath : string;

     MaximumComponentLength : DWORD;

     FileSystemFlags : DWORD;

     VolumeName : string;

    Begin

     result := false;

     DrivePath := Drive + ':\';

     if GetDriveType(PChar(DrivePath)) <> DRIVE_CDROM then exit;

     SetLength(VolumeName, 64);

     GetVolumeInformation(PChar(DrivePath),PChar(VolumeName), Length(VolumeName),nil,MaximumComponentLength,FileSystemFlags,nil,0);

     if lStrCmp(PChar(VolumeName),'Audio CD') = 0 then result := true;

    end;


    function PlayAudioCD(Drive : char) : bool;

    var mp : TMediaPlayer;

    begin

     result := false;

     Application.ProcessMessages;

     if not IsAudioCD(Drive) then exit;

     mp := TMediaPlayer.Create(nil);

     mp.Visible := false;

     mp.Parent := Application.MainForm;

     mp.Shareable := true;

     mp.DeviceType := dtCDAudio;

     mp.FileName := Drive + ':';

     mp.Shareable := true;

     mp.Open;

     Application.ProcessMessages;

     mp.Play;

     Application.ProcessMessages;

     mp.Close;

     Application.ProcessMessages;

     mp.free;

     result := true;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     if not PlayAudioCD('D') then ShowMessage('Not an Audio CD');

    end;


    Вопрос:

    Как узнать есть ли у мыши колесико?

    Ответ:

    Свойство "WheelPresent" глобального обьекта "mouse". 


    Вопрос:

    События KeyPress и KeyDown не вызываются для клавиши Tab — как определить, что она была нажата?

    Ответ:

    На уровне формы клавиша tab обычно обрабатывается Windows. В примере создается обработчик события CM_Dialog для перехвата Dialog keys.

    Пример:

    type TForm1 = class(TForm)

    private

     procedure CMDialogKey( Var msg: TCMDialogKey ); message CM_DIALOGKEY;

    end;

    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.CMDialogKey(var msg: TCMDialogKey);

    begin

     if msg.Charcode <> VK_TAB then inherited;

    end;


    procedure TForm1.FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);

    begin

     if Key = VK_TAB then Form1.Caption := 'Tab Key Down!';

    end;


    Вопрос:

    В чем отличие между Create(Self) и Create(Application)?

    Ответ:

    Self может быть использовано только в методе класса, и ссылается на текущий экземпляр класса. Таким образом "Self" в методе класса TForm1 ссылается на текущий экземпляр TForm1. При создании компонента Вы передаете его владельца (owner) в конструктор. При уничтожении формы или компонента автоматически уничтожаются и все компоненты владельцем которого она является. Таким образом если при создании формы передать в качестве владельца Application эта форма будет автоматически уничтожена при уничтожении Application. Если же при создании формы передать в качестве владельца другую форму, вновь созданная форма будет автоматически уничтоженн при уничтожении формы-владельца.


    Вопрос:

    Как во время выполнения определить поддерживает ли обьект заданное свойство?

    Ответ:

    function HasProperty(Obj : TObject; Prop : string) : PPropInfo;

    begin

     Result := GetPropInfo(Obj.ClassInfo, Prop);

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    var p : pointer;

    begin

     p := HasProperty(Button1, 'Color');

     if p <> nil then SetOrdProp(Button1, p, clRed)

     else ShowMessage('Button has no color property');

     p := HasProperty(Label1, 'Color');

     if p <> nil then SetOrdProp(Label1, p, clRed)

     else ShowMessage('Label has no color property');

     p := HasProperty(Label1.Font, 'Color');

     if p <> nil then SetOrdProp(Label1.Font.Color, p, clBlue)

     else ShowMessage('Label.Font has no color property');

    end;


    Вопрос:

    Как при проигрывании музыки с Audio CD показывать сколько прошло минут и секунд?

    Ответ:

    В примере время выводится по таймеру.

    Пример:

    uses MMSystem;

    procedure TForm1.Timer1Timer(Sender: TObject);

    var

     Trk : Word;

     Min : Word;

     Sec : Word;

    begin

     with MediaPlayer1 do begin

      Trk := MCI_TMSF_TRACK(Position);

      Min := MCI_TMSF_MINUTE(Position);

      Sec := MCI_TMSF_SECOND(Position);

      Label1.Caption := Format('%.2d',[Trk]);

      Label2.Caption := Format('%.2d:%.2d',[Min,Sec]);

     end;

    end;


    Вопрос:

    Можно ли рисовать на рамке формы?

    Ответ:

    Обрабатывайте событие WM_NCPAINT. В примере рамка обводится красной линией толщиной в 1 пиксел.

    Пример:

    type TForm1 = class(TForm)

    private

     {Private declarations}

     procedure WMNCPaint(var Msg : TWMNCPaint); message WM_NCPAINT;

    public

     {Public declarations}

    end;

    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.WMNCPaint(var Msg: TWMNCPaint);

    var

     dc : hDc;

     Pen : hPen;

     OldPen : hPen;

     OldBrush : hBrush;

    begin

     inherited;

     dc := GetWindowDC(Handle);

     msg.Result := 1;

     Pen := CreatePen(PS_SOLID, 1, RGB(255, 0, 0));

     OldPen := SelectObject(dc, Pen);

     OldBrush := SelectObject(dc, GetStockObject(NULL_BRUSH));

     Rectangle(dc, 0,0, Form1.Width, Form1.Height);

     SelectObject(dc, OldBrush);

     SelectObject(dc, OldPen);

     DeleteObject(Pen);

     ReleaseDC(Handle, Canvas.Handle);

    end;


    Вопрос:

    Как выполнить какой-то процесс тогда, когда пользователь не работает с моим приложением?

    Ответ:

    Создайте процедуру, которая будет вызываться при событии Application.OnIdle.

    Обьявим процедуру:

    {Private declarations}

    procedure IdleEventHandler(Sender: TObject; var Done: Boolean);

    В разделе implementation опишем процедуру:

    procedure TForm1.IdleEventHandler(Sender: TObject; var Done: Boolean);

    begin

     {Do a small bit of work here}

     Done := false;

    end;

    В методе Form'ы OnCreate — укажем что наша процедура вызывается на событии Application.OnIdle.

    Application.OnIdle := IdleEventHandler;

    Событие OnIdle возникает один раз — когда приложение переходит в режим "безделья" (idle). Если в обработчике переменной Done присвоить False событие будет вызываться вновь и вновь, до тех пор пока приложение "бездельничает" и переменной Done не присвоенно значение True.


    Вопрос:

    При перемещении фокуса ввода клавишей Tab чтобы переместить его в RadioGroup нужно нажать клавишу Tab дважды если какой нибудь пункт RadioGroup уже выбран, но только один раз если не выбран. Можно ли сделать поведение RadioGroup логичным?

    Ответ:

    Установка свойства RadioGroup'ы TabStop в false должна решить эту проблему — поскольку клавиша tab будет продолжать работать — перемещаясь сразу на выделенный пункт RadioGroup.


    Вопрос:

    Как разместить маленькие картинки в компоненте TPopUpMenu?

    Ответ:

    В приведенном примере показано как это сделать с использованием функции Windows API SetMenuItemBitmaps(). Эта функция получает handle popup menu, позицию строчки меню куда будет помещена картинка, и два дескриптора (handles) на две картинки (одна из них — картинка которая будет показана когда строка меню доступна, вторая — когда строка меню недоступна).

    type TForm1 = class(TForm)

     PopupMenu1: TPopupMenu;

     Pop11: TMenuItem;

     Pop21: TMenuItem;

     Pop31: TMenuItem;

     procedure FormCreate(Sender: TObject);

     procedure FormDestroy(Sender: TObject);

     procedure FormMouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    private

     {Private declarations}

     bmUnChecked : TBitmap;

     bmChecked : TBitmap;

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.FormCreate(Sender: TObject);

    begin

     bmUnChecked := TBitmap.Create;

     bmUnChecked.LoadFromFile('C:\Program Files\Borland\BitBtns\ALARMRNG.BMP');

     bmChecked := TBitmap.Create;

     bmChecked.LoadFromFile('C:\Program Files\Borland\BitBtns\CHECK.BMP');

     {Add the bitmaps to the item at index 1 in PopUpMenu}

     SetMenuItemBitmaps(PopUpMenu1.Handle,1,MF_BYPOSITION,BmUnChecked.Handle, BmChecked.Handle);

    end;


    procedure TForm1.FormDestroy(Sender: TObject);

    begin

     bmUnChecked.Free;

     bmChecked.Free;

    end;


    procedure TForm1.FormMouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    var pt : TPoint;

    begin

     pt := ClientToScreen(Point(x, y));

     PopUpMenu1.Popup(pt.x, pt.y);

    end;


    Вопрос:

    Как узнать число кадров AVI файла, и выяснить как долго будет проигрываться этот файл?

    Ответ:

    В приведенном примере указано как получить эту информацию.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     MediaPlayer1.TimeFormat := tfFrames;

     ShowMessage('Number of frames = ' + IntToStr(MediaPlayer1.Length));

     MediaPlayer1.TimeFormat := tfMilliseconds;

     ShowMessage('Number of milliseconds = ' + IntToStr(MediaPlayer1.Length));

    end;


    Вопрос:

    Как изменить число фиксированных колонок в TDbGrid?

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     TStringGrid(DbGrid1).FixedCols := 2;

    end;


    Вопрос:

    Некоторые компоненты баз данных (и среди них TDBGrid) никак не меняют визуальных свойств, когда к ним отключен доступ (disabled). Как это изменить програмно?

    Ответ:

    Ниже приведен пример, меняющий цвет шрифта на clGray, когда доступ к элементу управления (в данном случае TDBGrid) запрещен (disabled).

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     DbGrid1.Enabled := false;

     DbGrid1.Font.Color := clGray;

    end;


    procedure TForm1.Button2Click(Sender: TObject);

    begin

     DbGrid1.Enabled := true;

     DbGrid1.Font.Color := clBlack;

    end;


    Вопрос:

    Как определить нажаты ли клавиши Shift, Alt, or Ctrl в какой-либо момент времени?

    Ответ:

    В приведенном примере показано как определить нажата ли клавиша Shift при выборе строчки меню. Пример также содержит функции проверки состояния клавиш Alt, Ctrl.

    Пример:

    function CtrlDown : Boolean;

    var State : TKeyboardState;

    begin

     GetKeyboardState(State);

     Result := ((State[vk_Control] And 128) <> 0);

    end;


    function ShiftDown : Boolean;

    var State : TKeyboardState;

    begin

     GetKeyboardState(State);

     Result := ((State[vk_Shift] and 128) <> 0);

    end;


    function AltDown : Boolean;

    var State : TKeyboardState;

    begin

     GetKeyboardState(State);

     Result := ((State[vk_Menu] and 128) <> 0);

    end;


    procedure TForm1.MenuItem12Click(Sender: TObject);

    begin

     if ShiftDown then Form1.Caption := 'Shift'

     else Form1.Caption := '';

    end;


    Вопрос:

    Как изменить шрифта hint'а?

    Ответ:

    В примере перехватывается событие Application.OnShowHint и изменяется шрифт Hint'а.

    Пример:

    type TForm1 = class(TForm)

     procedure FormCreate(Sender: TObject);

    private

     {Private declarations}

    public

     procedure MyShowHint(var HintStr: string; var CanShow: Boolean;var HintInfo: THintInfo);

     {Public declarations}

    end;

    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.MyShowHint(var HintStr: string; var CanShow: Boolean; var HintInfo: THintInfo);

    var i : integer;

    begin

     for i := 0 to Application.ComponentCount - 1 do if Application.Components[i] is THintWindow then with THintWindow(Application.Components[i]).Canvas do begin

      Font.Name:= 'Arial';

      Font.Size:= 18;

      Font.Style:= [fsBold];

      HintInfo.HintColor:= clWhite;

     end;

    end;


    procedure TForm1.FormCreate(Sender: TObject);

    begin

     Application.OnShowHint := MyShowHint;

    end;


    Вопрос:

    Есть ли в Delphi эквивалент функции SendKeys Visual Basic'а?

    Ответ:

    Ниже приведена процедура, позволяющаю отправлять нажатия в любой элемент управления (window control), способный принимать ввод с клавиатуры. Вы можете использовать эту технику чтобы включать клавиши NumLock, CapsLock и ScrollLock под Windows NT. Та же техника работает и под Windows 95 для CapsLock и ScrollLock но не работает для клавиши NumLock.

    Обратите внимание, что приведены четыре процедуры: SimulateKeyDown() — эмулировать нажатие клавиши (без отпускания), SimulateKeyUp() — эмулировать отпускание клавиши, SimulateKeystroke() — эмулировать удар по клавише (нажатие и отпускание) и SendKeys(), позволяющие Вам гибко контролировать посылаемые сообщения клавиатуры.

    SimulateKeyDown(), SimulateKeyUp() и SimulateKeystroke() получают коды виртуальных клавиш (virtural key) (вроде VK_F1). Процедура SimulateKeystroke() получает дополнительный параметр, полезный при эмуляции нажатия PrintScreen. Когда этот параметр равен нулю весь экран будет скопирован в буфер обмена (clipboard). Если дополнительный параметр равен 1 будет скопированно только активное окно.

    Четыре метода "button click" демонстрируют использование: ButtonClick1 — включает capslock, ButtonClick2 — перехватывает весь экран в буфер обмена (clipboard), ButtonClick3 — перехватывает активное окно в буфер обмена (clipboard). ButtonClick4 — устанавливает фокус в Edit и отправляет в него строку.

    Пример:

    procedure SimulateKeyDown(Key : byte);

    begin

     keybd_event(Key, 0, 0, 0);

    end;


    procedure SimulateKeyUp(Key : byte);

    begin

     keybd_event(Key, 0, KEYEVENTF_KEYUP, 0);

    end;


    procedure SimulateKeystroke(Key : byte; extra : DWORD);

    begin

     keybd_event(Key,extra,0,0);

     keybd_event(Key,extra,KEYEVENTF_KEYUP,0);

    end;


    procedure SendKeys(s : string);

    var

     i : integer;

     flag : bool;

     w : word;

    begin

     {Get the state of the caps lock key}

     flag := not GetKeyState(VK_CAPITAL) and 1 = 0;

     {If the caps lock key is on then turn it off}

     if flag then SimulateKeystroke(VK_CAPITAL, 0);

     for i := 1 to Length(s) do begin

      w := VkKeyScan(s[i]);

      {If there is not an error in the key translation}

      if ((HiByte(w) <> $FF) and (LoByte(w) <> $FF)) then begin

       {If the key requires the shift key down - hold it down}

       if HiByte(w) and 1 = 1 then SimulateKeyDown(VK_SHIFT);

       {Send the VK_KEY}

       SimulateKeystroke(LoByte(w), 0);

       {If the key required the shift key down - release it}

       if HiByte(w) and 1 = 1 then SimulateKeyUp(VK_SHIFT);

      end;

     end;

     {if the caps lock key was on at start, turn it back on}

     if flag then SimulateKeystroke(VK_CAPITAL, 0);

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     {Toggle the cap lock}

     SimulateKeystroke(VK_CAPITAL, 0);

    end;


    procedure TForm1.Button2Click(Sender: TObject);

    begin

     {Capture the entire screen to the clipboard}

     {by simulating pressing the PrintScreen key}

     SimulateKeystroke(VK_SNAPSHOT, 0);

    end;


    procedure TForm1.Button3Click(Sender: TObject);

    begin

     {Capture the active window to the clipboard}

     {by simulating pressing the PrintScreen key}

     SimulateKeystroke(VK_SNAPSHOT, 1);

    end;


    procedure TForm1.Button4Click(Sender: TObject);

    begin

     {Set the focus to a window (edit control) and send it a string}

     Application.ProcessMessages;

     Edit1.SetFocus;

     SendKeys('Delphi Is RAD!');

    end;


    Вопрос:

    Я загружаю TImageList динамически. Как сделать картинки из TImageList прозрачными?

    Ответ:

    См. ответ.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     bm : TBitmap;

     il : TImageList;

    begin

     bm := TBitmap.Create;

     bm.LoadFromFile('C:\DownLoad\TEST.BMP');

     il := TImageList.CreateSize(bm.Width,bm.Height);

     il.DrawingStyle := dsTransparent;

     il.Masked := true;

     il.AddMasked(bm, clRed);

     il.Draw(Form1.Canvas, 0, 0, 0);

     bm.Free;

     il.Free;

    end;


    Вопрос:

    Как заставить TMediaPlayer проигрывать одно и тоже бесконечно? AVI например?

    Ответ:

    В примере AVI файл проигрывается снова и снова — используем событие MediaPlayer'а Notify

    Пример:

    procedure TForm1.MediaPlayer1Notify(Sender: TObject);

    begin

     with MediaPlayer1 do if NotifyValue = nvSuccessful then begin

      Notify := True;

      Play;

     end;

    end;


    Вопрос:

    При выполнении диалога FontDialog со свойством Device равным fdBoth or fdPrinter, появляется ошибка "There are no fonts installed".

    Ответ:

    Эти установки должны показать шрифты совместимые либо с принтером либо с экраном. В примере диалог Windows ChooseFont вызывается напрямую чтобы показать список шрифтов, совместимых одновременно и с экраном и с принтером.

    Пример:

    uses Printers, CommDlg;


    procedure TForm1.Button1Click(Sender: TObject);

    var

     cf : TChooseFont;

     lf : TLogFont;

     tf : TFont;

    begin

     if PrintDialog1.Execute then begin

      GetObject(Form1.Canvas.Font.Handle, sizeof(lf),@lf);

      FillChar(cf, sizeof(cf), #0);

      cf.lStructSize := sizeof(cf);

      cf.hWndOwner := Form1.Handle;

      cf.hdc := Printer.Handle;

      cf.lpLogFont := @lf;

      cf.iPointSize := Form1.Canvas.Font.Size * 10;

      cf.Flags := CF_BOTH or CF_INITTOLOGFONTSTRUCT or CF_EFFECTS or CF_SCALABLEONLY or CF_WYSIWYG;

      cf.rgbColors := Form1.Canvas.Font.Color;

      if ChooseFont(cf) <> false then begin

       tf := TFont.Create;

       tf.Handle := CreateFontIndirect(lf);

       tf.COlor := cf.RgbColors;

       Form1.Canvas.Font.Assign(tf);

       tf.Free;

       Form1.Canvas.TextOut(10, 10, 'Test');

      end;

     end;

    end;


    Вопрос:

    Как сменить дисковод, откуда MediaPlayer проигрывает аудио CD?

    Ответ:

    См. пример.

    Пример:

    MediaPlayer1.FileName := 'E:';


    Вопрос:

    Как убрать кнопку с названием моей программы из Панели Задач(Taskbar)?

    Ответ:

    Отредактируйте файл-проекта (View→Project Source). Добавьте модуль Windows в раздел uses. Application.ShowMainForm := False; в строку после "Application.Initialize;". Добавьте ShowWindow(Application.Handle, SW_HIDE); в строку перед "Application.Run;"

    Ваш файл проекта должен выглядеть приблизительно так:

    program Project1;

    uses Windows, Forms, Unit1 in 'Unit1.pas' {Form1}, Unit2 in 'Unit2.pas' {Form2};

    {$R *.RES}

    begin

     Application.Initialize;

     Application.ShowMainForm := False;

     Application.CreateForm(TForm1, Form1);

     Application.CreateForm(TForm2, Form2);

     ShowWindow(Application.Handle, SW_HIDE);

     Application.Run;

    end.

    В разделе "initialization" (в самом низу) каждого unit'а добавьте

    begin

     ShowWindow(Application.Handle, SW_HIDE);

    end.


    Вопрос:

    Как преобразовать цвета в строку — название цвета VCL?

    Ответ:

    Модуль graphics.pas содержит функцию ColorToString() которое преобразует допустимое значение TColor в его строковое представление используя либо константу — название цвета (по возможности) либо шестнадцатиричную строку. Обратная функция — StringToColor()

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Memo1.Lines.Add(ColorToString(clRed));

     Memo1.Lines.Add(IntToStr(StringToColor('clRed')));

    end;


    Вопрос:

    При показе максимизированной формы она перекрывает task bar и не выравнивается по верху экрана. В чем тут дело?

    Ответ:

    Это может произойти когда свойство position формы установленно в poScreenCenter. Установите position = poDefault.


    Вопрос:

    Как заставить TEdit не 'пикать' при нажатии недопустимых клавиш?

    Ответ:

    Перехватите событие KeyPress и установите key = #0 для недопустимых клавиш.

    Пример:

    procedure TForm1.Edit1KeyPress(Sender: TObject; var Key: Char);

    begin

     if ((UpCase(Key) < 'A') or (UpCase(Key) > 'Z')) then Key := #0;

    end;


    Вопрос:

    Как получить число и список всех компонентов, расположенных на TNoteBook?

    Ответ:

    В примере список выводится на Listbox.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     n: integer;

     p: integer;

    begin

     ListBox1.Clear;

     with Notebook1 do begin

      for n := 0 to ControlCount - 1 do begin

       with TPage(Controls[n]) do begin

        ListBox1.Items.Add('Notebook Page: ' + TPage(Notebook1.Controls[n]).Caption);

        for p := 0 to ControlCount - 1 do ListBox1.Items.Add(Controls[p].Name);

        ListBox1.Items.Add(EmptyStr);

       end;

      end;

     end;

    end;


    Вопрос:

    Я хочу вставить escape code в строку при использовании функции Format(). Например, я хочу создать строку, содержащую символ табуляции. В "C" я бы написал что-то вроде sprintf(buffer, "%s\t%s", str);. А как это будет на Pascal'e?

    Ответ:

    Функция Format Pascal'я не использует escape codes. Вместо этого нужно вставить в строку действительное значение символа в кодировке ASCII.

    Пример:

    Buffer := Format('%s'#9'%s', [Str1, Str2]);

    ShowMessage(Format('%s'#9'%s', ['Column1', 'Column2']));


    Вопрос:

    Как показать первый кадр AVI-файла?

    Ответ:

    См. пример.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Application.ProcessMessages;

     MediaPlayer1.Open;

     Application.ProcessMessages;

     MediaPlayer1.Step;

     Application.ProcessMessages;

     MediaPlayer1.Previous;

    end;


    Вопрос:

    Когда пользователь щелкает по listview, он переходит в режим редактирования. Как перевести его в режим редактирования по нажатию клавиши (например F2)?

    Ответ:

    Перехватите F2 на событии keydown.

    Пример:

    procedure TForm1.ListView1KeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);

    begin

     if Ord(Key) = VK_F2 then ListView1.Selected.EditCaption;

    end;


    Вопрос:

    Когда я добавляю обьект в список TStrings как мне его потом уничтожить?

    Ответ:

    Просто вызовите метод free этого обьекта.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    var Icon: TIcon;

    begin

     Icon := TIcon.Create;

     Icon.LoadFromFile('C:\Program Files\BorlandImages\CONSTRUC.ICO');

     ListBox1.Items.AddObject('Item 0', Icon);

    end;


    procedure TForm1.FormDestroy(Sender: TObject);

    begin

     ListBox1.Items.Objects[0].Free;

    end;


    Вопрос:

    Вместо печати графики я хочу использовать резидентный шрифт принтера. Как?

    Ответ:

    Используте функцию Windows API — GetStockObject() чтобы получить дескриптор (handle) шрифта по умолчанию устройства (DEVICE_DEFAULT_FONT) и передайте его Printer.Font.Handle.

    Пример:

    uses Printers;

    procedure TForm1.Button1Click(Sender: TObject);

    var

     tm : TTextMetric;

     i : integer;

    begin

     if PrintDialog1.Execute then begin

      Printer.BeginDoc;

      Printer.Canvas.Font.Handle := GetStockObject(DEVICE_DEFAULT_FONT);

      GetTextMetrics(Printer.Canvas.Handle, tm);

      for i := 1 to 10 do begin

       Printer.Canvas.TextOut(100,i * tm.tmHeight + tm.tmExternalLeading,'Test');

      end;

      Printer.EndDoc;

     end;

    end;


    Вопрос:

    Мне нужно программно установить некоторые файлы с установочного диска Windows. На многих компьютерах CAB-файлы установки Windows находятся в каком-то каталоге на жестком диске, на других — Windows был установлен с CD. Как узнать откуда была установленна Windows?

    Ответ:

    Эту информацию можно получить из реестра.

    Пример:

    uses Registry;

    procedure TForm1.Button1Click(Sender: TObject);

    var reg: TRegistry;

    begin

     reg := TRegistry.Create;

     reg.RootKey := HKEY_LOCAL_MACHINE;

     reg.OpenKey('Software\Microsoft\Windows\CurrentVersion\SETUP',false);

     ShowMessage(reg.ReadString('SourcePath'));

     reg.CloseKey;

     reg.free;

    end;


    Вопрос:

    Как получить строку сообщения об ошибке Windows код которой получен функцией GetLastError?

    Ответ:

    Функция RTL SysErrorMessage(GetLastError).

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     {Cause a Windows system error message to be logged}

     ShowMessage(IntToStr(lStrLen(nil)));

     ShowMessage(SysErrorMessage(GetLastError));

    end;


    Вопрос:

    Как заставить Delphi выполнять еще более строгую проверка типов? Например — я создаю пользовательский тип, унаследованный от double и могу передавать его любым функциям, принимающим параметр типа double. Как заставить компилятор проводить более строгую проверку типов и выдавать предупреждение в таких случаях?

    Ответ:

    См. ответ.

    Пример:

    type TStrongType = type Double;

    type TWeakType = Double;


    procedure AddWeakType(var d : TWeakType);

    begin

     d := d + 1;

    end;


    procedure AddStrongType(var d : TStrongType);

    begin

     d := d + 1;

    end;


    procedure AddDoubleType(var d : Double);

    begin

     d := d + 1;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    var

     d : Double;

     s : TStrongType;

     w : TWeakType;

    begin

     AddDoubleType(d); {compiles fine}

     AddDoubleType(w); {compiles fine}

     AddDoubleType(s); {<- compile error}

     AddDoubleType(double(s)); {compiles fine}

     AddWeakType(d); {compiles fine}

     AddWeakType(w); {compiles fine}

     AddWeakType(s); {<- compile error}

     AddWeakType(TWeakType(s)); {compiles fine}

     AddStrongType(d); {<- compile error}

     AddStrongType(TStrongType(d)); {compiles fine}

     AddStrongType(w); {<- compile error}

     AddStrongType(TStrongType(w)); {compiles fine}

     AddStrongType(s); {compiles fine}

    end;


    Вопрос:

    Где в Delphi обьявленны VK_Key для A-Z и 0-9?

    Ответ:

    Они не обьявлены в Delphi поскольку они просто могуть быть заменены буквами. VK_0 до VK_9 то же что и ASCII '0' до '9' ($30 – $39), VK_A до VK_Z то же что и ASCII 'A' до 'Z' ($41 – $5A).


    Вопрос:

    Как изменить оконную процедуру для TForm?

    Ответ:

    Переопределите в подклассе TForm оконную процедуру WinProc класса. В примере оконная процедура переопределяется для того чтобы реагировать на сообщение WM_CANCELMODE, показывающее, что выполняется messagebox или какой-либо еще диалог.

    Пример:

    type TForm1 = class(TForm)

     Button1: TButton;

     procedure WndProc (var Message: TMessage); override;

     procedure Button1Click(Sender: TObject);

    private

     {Private declarations}

    public

     {Public declarations}

    end;


    var Form1: TForm1;

    implementation

    {$R *.DFM}


    procedure TForm1.WndProc(var Message: TMessage);

    begin

     if Message.Msg = WM_CANCELMODE then begin

      Form1.Caption := 'A dialog or message box has popped up';

     end else inherited // <- остальное сделает родительская процедура

    end;


    procedure TForm1.Button1Click(Sender: TObject);

     begin ShowMessage('Test Message');

    end;


    Вопрос:

    Как узнать размеры TComboBox с показанным выпадающим списком до показа списка?

    Ответ:

    На событии FormShow пошлите сообщение CB_SHOWDROPDOWN в ComboBox дважды — один раз чтобы заставить список выпасть, второй — чтобы убрать его. Затем пошлите сообщение CB_GETDROPPEDCONTROLRECT, передав в качестве параметра адрес TRect. TRect будет содержать экранные кординаты прямоугольника описывающего ComboBox вместе с выпавшим списком. Затем Вы можете вызвать ScreenToClient чтобы преобразовать экранные кординаты в координаты клиентской области окна.

    Пример:

    var R : TRect;

    procedure TForm1.FormShow(Sender: TObject);

    var T : TPoint;

    begin

     SendMessage(ComboBox1.Handle, CB_SHOWDROPDOWN, 1, 0);

     SendMessage(ComboBox1.Handle, CB_SHOWDROPDOWN, 0, 0);

     SendMessage(ComboBox1.Handle, CB_GETDROPPEDCONTROLRECT, 0, LongInt(@r));

     t := ScreenToClient(Point(r.Left, r.Top));

     r.Left := t.x;

     r.Top := t.y;

     t := ScreenToClient(Point(r.Right, r.Bottom));

     r.Right := t.x;

     r.Bottom := t.y;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Form1.Canvas.Rectangle(r.Left, r.Top, r.Right, r.Bottom );

    end;


    Вопрос:

    Я хочу создать в своей программе меню "а ля Дельфи 4". Как это сделать?

    Ответ:

    1. Разместите на форме TControlBar. (закладка Additional) Установите Align = Client.

    2. Разместите TToolBar (закладка Win32) внутри TControlBar.

    3. Установите в True свойства Flat и ShowCaptions этого TToolBar.

    4. Создайте на TToolBar столько TToolButtons сколько Вам нужно. (щелкнув по TToolBar правой кнопкой и выбрав NewButton)

    5. Установите свойство Grouped = True для всех TToolButtons. Это позволит меню выпадать при перемещении курсора между главными пунктами меню (если меню уже показано).

    6. Разместите на фоме TMainMenu и убедитесь, что оно НЕ присоединено как меню главной формы. (посмотрите свойство Menu формы).

    7. Создайте все пункты меню (щелкнув по TMainMenu кнопкой и выбрав Menu Designer)

    8. Для каждой TToolButton установите ее MenuItem равным соответсвующему пункту TMainMenu.


    Вопрос:

    Как добиться того, чтобы TMemo и TEdit имели работали не только в режиме вставки символов, но и в режиме замены?

    Ответ:

    Элементы управления Windows TEdit и TMemo не имеют режима замены. Однако этот режим можно эмулировать установив свойство SelLength edit'а или memo в 1 при обработке события KeyPress. Это заставит его перезаписывать символ в текущей позиции курсора. В примере этот способ используется для TMemo. Режим вставка/замена переключается клавишей "Insert".

    Пример:

    type TForm1 = class(TForm)

     Memo1: TMemo;

     procedure Memo1KeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);

     procedure Memo1KeyPress(Sender: TObject; var Key: Char);

    private

     {Private declarations}

     InsertOn : bool;

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation


    {$R *.DFM}


    procedure TForm1.Memo1KeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);

    begin

     if (Key = VK_INSERT) and (Shift = []) then InsertOn := not InsertOn;

    end;


    procedure TForm1.Memo1KeyPress(Sender: TObject; var Key: Char);

    begin

     if ((Memo1.SelLength = 0) and (not InsertOn)) then Memo1.SelLength := 1;

    end;


    Вопрос:

    Как отправить сообщение сразу всем элементам управления формы?

    Ответ:

    Можно использовать Screen.Forms[i].BroadCast(msg); где [i] — индекс той формы, которой Вы хотите переслать сообщение. BroadCast работает со всеми компонентами, потомками TWinControls и отправляет сообщение всем дочерним компонентам из массива Controls. Если один из дочерних компонентов обрабатывает это сообщение и устанавливает Msg.Result в ненулевое значение — дальнейшая рассылка сообщения останавливается.


    Вопрос:

    При попытке присвоить значение свойству "selected" ListBox'а вырабатывается exception "Index is out of bounds". В чем тут дело и как присвоить значение свойству selected?

    Ответ:

    Свойство "selected" компонента ТListBox может быть использованно только если свойство MultiSelect установленно в True. Если Вы работаете с ListBox'ом у которого MultiSelect=false то используйте свойство ItemIndex.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     ListBox1.Items.Add('1');

     ListBox1.Items.Add('2');

     {This will fail on a single selection ListBox}

     // ListBox1.Selected[1] := true;

     ListBox1.ItemIndex := 1; {This is ok}

    end;


    Вопрос:

    Как ограничить длину текста, вводимого в TEdit, так чтобы ширина текста не превышала ширину TEdit'а?

    Ответ:

    В примере приведено два способа ограничить длину текста в TEdit так чтобы она не превышала ширину клиентской области окна TEdit'а и не появлялась прокрутка текста. Первый способ устанавливает свойство TEdit'а MaxLength равным числу букв "W", которые поместятся в TEdit. "W" выбрана потому, что является, наверное, самой широкой буквой в любом шрифте. Этот метод сносно работает для шрифтов с фиксированной шириной букв, но для шрифтов с переменной шириной букв вряд ли сгодится. Второй способ перхватывает событие KeyPress TEdit'а и измеряет ширину уже введенного текста и ширину нового символа. Если ширина больше чем клиентская область TEdit'а новый символ отбрасывается и вызывается MessageBeep.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    var

     cRect : TRect;

     bm : TBitmap;

     s : string;

    begin

     Windows.GetClientRect(Edit1.Handle, cRect);

     bm := TBitmap.Create;

     bm.Width := cRect.Right;

     bm.Height := cRect.Bottom;

     bm.Canvas.Font := Edit1.Font;

     s := 'W';

     while bm.Canvas.TextWidth(s) < CRect.Right do s := s + 'W';

     if length(s) > 1 then begin

      Delete(s, 1, 1);

      Edit1.MaxLength := Length(s);

     end;

    end;


    {Другой вариант}

    procedure TForm1.Edit1KeyPress(Sender: TObject; var Key: Char);

    var

     cRect : TRect;

     bm : TBitmap;

    begin

     if ((Ord(Key) <> VK_TAB) and (Ord(Key) <> VK_RETURN) and (Ord(Key) <> VK_LEFT) and (Ord(Key) <> VK_BACK)) then begin

      Windows.GetClientRect(Edit1.Handle, cRect);

      bm := TBitmap.Create;

      bm.Width := cRect.Right;

      bm.Height := cRect.Bottom;

      bm.Canvas.Font := Edit1.Font;

      if bm.Canvas.TextWidth(Edit1.Text + Key) > CRect.Right then begin

       Key := #0;

       MessageBeep(-1);

      end;

      bm.Free;

     end;

    end;


    Вопрос:

    Как сохранить обьект TFont в реестре/ini/файле/таблице базы данных?

    Ответ:

    Нужно сохранять атрибуты шрифта (имя, размер и т.п.) а не сам обьект TFont. После считывания этой информации следует проверить существует ли такой шрифт, прежде чем его использовать. Чтобы не показаться голословным дополню ответ Borland'а своим примером сохранения/чтения шрифта в/из реестра

    Uses ... Registry;


    procedure SaveFontToRegistry(Font : TFont; SubKey : String);

    Var R :

     TRegistry;

     FontStyleInt : byte;

     FS : TFontStyles;

    begin

     R:=TRegistry.Create;

     try

      FS:=Font.Style;

      Move(FS,FontStyleInt,1);

      R.OpenKey(SubKey,True);

      R.WriteString('Font Name',Font.Name);

      R.WriteInteger('Color',Font.Color);

      R.WriteInteger('CharSet',Font.Charset);

      R.WriteInteger('Size',Font.Size);

      R.WriteInteger('Style',FontStyleInt);

     finally

      R.Free;

     end;

    end;


    function ReadFontFromRegistry(Font : TFont; SubKey : String) : boolean;

    Var

     R : TRegistry;

     FontStyleInt : byte;

     FS : TFontStyles;

    begin

     R:=TRegistry.Create;

     try

      result:=R.OpenKey(SubKey,false);

      if not result then exit;

      Font.Name:=R.ReadString('Font Name');

      Font.Color:=R.ReadInteger('Color');

      Font.Charset:=R.ReadInteger('CharSet');

      Font.Size:=R.ReadInteger('Size');

      FontStyleInt:=R.ReadInteger('Style');

      Move(FontStyleInt,FS,1);

      Font.Style:=FS;

     finally

      R.Free;

     end;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     If FontDialog1.Execute then begin

      SaveFontToRegistry(FontDialog1.Font,'Delphi Kingdom\Fonts');

     end;

    end;


    procedure TForm1.Button2Click(Sender: TObject);

    var NFont : TFont;

    begin

     NFont:=TFont.Create;

     if ReadFontFromRegistry(NFont,'Delphi Kingdom\Fonts') then begin

      //здесь добавить проверку - существует ли шрифт

      Label1.Font.Assign(NFont);

      NFont.Free;

     end;

    end;


    Вопрос:

    Как перемещать компонент мышкой во время работы программы "runtime"?

    Ответ:

    Перехватить событие OnMouseDown, запомнить x и y координты курсора мыши. Отслеживать движение мыши по событию OnMouseMove и перемещать компонент вслед за курсором мыши до тех пор пока не произойдет событие OnMouseUp. В примере показано перемещение компонента TButton. Перемещение начинается, когда пользователь "берет" TButton мышью, удерживая нажатой клавишу "Сontrol".

    Пример:

    type TForm1 = class(TForm)

     Button1: TButton;

     procedure Button1MouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

     procedure Button1MouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

     procedure Button1MouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    private

     {Private declarations}

    public

     {Public declarations}

     MouseDownSpot : TPoint;

     Capturing : bool;

    end;


    var Form1: TForm1;


    implementation


    {$R *.DFM}


    procedure TForm1.Button1MouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    begin

     if ssCtrl in Shift then begin

      SetCapture(Button1.Handle);

      Capturing := true;

      MouseDownSpot.X := x;

      MouseDownSpot.Y := Y;

     end;

    end;


    procedure TForm1.Button1MouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

    begin

     if Capturing then begin

      Button1.Left := Button1.Left - (MouseDownSpot.x - x);

      Button1.Top := Button1.Top - (MouseDownSpot.y - y);

     end;

    end;


    procedure TForm1.Button1MouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    begin

     if Capturing then begin

      ReleaseCapture;

      Capturing := false;

      Button1.Left := Button1.Left - (MouseDownSpot.x - x);

      Button1.Top := Button1.Top - (MouseDownSpot.y - y);

     end;

    end;


    Вопрос:

    При попытке создать обьект класса TPrinter (TPrinter.Create) я получаю exception. Почему?

    Ответ:

    В создании обьекта класса TPrinter с использованием TPrinter.Create нет необходимости, так как обьект класса TPrinter (называемый Printer) автоматически создается при использовании модуля Printers.

    Пример:

    uses Printers;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Printer.BeginDoc;

     Printer.Canvas.TextOut(100, 100, 'Hello World!');

     Printer.EndDoc;

    end;


    Вопрос:

    Как перехватить события в неклиентской области формы, в заголовке окна, например?

    Ответ:

    Создайте обработчик одного из сообщений WM_NC (non client — не клиентских) (посмотрите WM_NC в Windows API help). Пример показывает как перехватить вижение мыши во всей неклиенстской области окна (рамка и заголовок).

    Пример:

    unit Unit1;

    interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls;

    type TForm1 = class(TForm)

    private

     {Private declarations}

     procedure WMNCMOUSEMOVE(var Message: TMessage); message WM_NCMOUSEMOVE;

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation


    {$R *.DFM}


    procedure TForm1.WMNCMOUSEMOVE(var Message: TMessage);

    var s : string;

    begin

     case Message.wParam of

     HTERROR: s:= 'HTERROR';

     HTTRANSPARENT: s:= 'HTTRANSPARENT';

     HTNOWHERE: s:= 'HTNOWHERE';

     HTCLIENT: s:= 'HTCLIENT';

     HTCAPTION: s:= 'HTCAPTION';

     HTSYSMENU: s:= 'HTSYSMENU';

     HTSIZE: s:= 'HTSIZE';

     HTMENU: s:= 'HTMENU';

     HTHSCROLL: s:= 'HTHSCROLL';

     HTVSCROLL: s:= 'HTVSCROLL';

     HTMINBUTTON: s:= 'HTMINBUTTON';

     HTMAXBUTTON: s:= 'HTMAXBUTTON';

     HTLEFT: s:= 'HTLEFT';

     HTRIGHT: s:= 'HTRIGHT';

     HTTOP: s := 'HTTOP';

     HTTOPLEFT: s:= 'HTTOPLEFT';

     HTTOPRIGHT: s:= 'HTTOPRIGHT';

     HTBOTTOM: s:= 'HTBOTTOM';

     HTBOTTOMLEFT: s:= 'HTBOTTOMLEFT';

     HTBOTTOMRIGHT: s:= 'HTBOTTOMRIGHT';

     HTBORDER: s:= 'HTBORDER';

     HTOBJECT: s:= 'HTOBJECT';

     HTCLOSE: s:= 'HTCLOSE';

     HTHELP: s:= 'HTHELP';

     else s:= '';

     end;

     Form1.Caption := s;

     Message.Result := 0;

    end;


    end.


    Вопрос:

    При попытке использовать метод TCanvas.StretchDraw чтобы нарисовать иконку увеличенной ее размер не изменяется. Что делать?

    Ответ:

    Иконки всегда рисуются размером принятым в системе по умолчанию. Чтобы показать увеличенный вид иконки скопируйте ее на bitmap, а зате используйте метод TCanvas.StretchDraw.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var TheBitmap : TBitmap;

    begin

     TheBitmap := TBitmap.Create;

     TheBitmap.Width := Application.Icon.Width;

     TheBitmap.Height := Application.Icon.Height;

     TheBitmap.Canvas.Draw(0, 0, Application.Icon);

     Form1.Canvas.StretchDraw(Rect(0,0,TheBitmap.Width * 3,TheBitmap.Height * 3), TheBitmap);

     TheBitmap.Free;

    end;


    Вопрос:

    Можно ли сделать так чтобы TStringGrid автоматически изменял ширину колонок, чтобы вместить самую длинную строчку в колонке?

    Ответ: См. пример.

    Пример:

    procedure AutoSizeGridColumn(Grid : TStringGrid; column : integer);

    var

     i : integer;

     temp : integer;

     max : integer;

    begin

     max := 0;

     for i := 0 to (Grid.RowCount - 1) do begin

      temp := Grid.Canvas.TextWidth(grid.cells[column, i]);

      if temp > max then max := temp;

     end;

     Grid.ColWidths[column] := Max + Grid.GridLineWidth + 3;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     AutoSizeGridColumn(StringGrid1, 1);

    end;


    Вопрос:

    TTimer работает не достаточно точно. Как получить более высокую точность?

    Ответ:

    Таймер Windows не был создан с целью получения сверхточного хронометра. :-( Другими словами, когда Вы устанавливаете таймер на срабатывания каждые 1000 миллисекунд, он может срабатывать через интервал несколько больший чем 1000 миллисекунд. Значения меньше 55 миллисекунд никогда не будут срабатывать вовремя в Windows, поскольку это минимальная точность таймера. Можно проверять системное время и сравнивать его со временем предыдущего события таймера чтобы повысить точность.


    Вопрос:

    Как поместить JPEG-картинку в exe-файл и потом загрузить ее?

    Ответ:

    1) Создайте текстовый файл с расширением ".rc". Имя этого файла должно отличаться от имени файла-проекта или любого модуля проекта.

    Файл должен содержать строку вроде: MYJPEG JPEG C:\DownLoad\MY.JPG

    где:

    • "MYJPEG" — имя ресурса

    • "JPEG" — пользовательский тип ресурса

    • "C:\DownLoad\MY.JPG" — путь к JPEG файлу.

    Пусть например rc-файл называется "foo.rc"

    Запустите BRCC32.exe (Borland Resource CommandLine Compiler) — программа находится в каталоге Bin Delphi/C++ Builder'а — передав ей в качестве параметра полный путь к rc-файлу. В нашем примере:

    C:\DelphiPath\BIN\BRCC32.EXE C:\ProjectPath\FOO.RC

    Вы получите откомпилированный ресурс — файл с расширением ".res". (в нашем случае — foo.res). Далее добавьте ресурс к своему приложению.

    {Грузим ресурс}

    {$R FOO.RES}

    uses Jpeg;


    procedure LoadJPEGFromRes(TheJPEG : string; ThePicture : TPicture);

    var

     ResHandle : THandle;

     MemHandle : THandle;

     MemStream : TMemoryStream;

     ResPtr : PByte;

     ResSize : Longint;

     JPEGImage : TJPEGImage;

     begin ResHandle := FindResource(hInstance, PChar(TheJPEG), 'JPEG');

     MemHandle := LoadResource(hInstance, ResHandle);

     ResPtr := LockResource(MemHandle);

     MemStream := TMemoryStream.Create;

     JPEGImage := TJPEGImage.Create;

     ResSize := SizeOfResource(hInstance, ResHandle);

     MemStream.SetSize(ResSize);

     MemStream.Write(ResPtr^, ResSize);

     FreeResource(MemHandle);

     MemStream.Seek(0, 0);

     JPEGImage.LoadFromStream(MemStream);

     ThePicture.Assign(JPEGImage);

     JPEGImage.Free;

     MemStream.Free;

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     LoadJPEGFromRes('MYJPEG', Image1.Picture);

    end;


    Вопрос:

    Как перехватить сообщения прокрутки в TScrollBox?

    Ответ:

    Следующий пример перехватывает сообщения о прокрутке компонента TScrollBox и синхронизирует обе линейки прокрутки. Сообщения прокрутки перехватываются с помощью переопределения оконной процедуры (WinProc) ScrollBox'а.

    Пример:

    type

    {$IFDEF WIN32}

    WParameter = LongInt;

    {$ELSE}

    WParameter = Word;

    {$ENDIF}

    LParameter = LongInt;


    {Declare a variable to hold the window procedure we are replacing}

    var OldWindowProc : Pointer;


    function NewWindowProc(WindowHandle : hWnd; TheMessage : WParameter; ParamW : WParameter; ParamL : LParameter) : LongInt

    {$IFDEF WIN32} stdcall; {$ELSE} ; export; {$ENDIF}

    var

     TheRangeMin : integer;

     TheRangeMax : integer;

     TheRange : integer;

    begin

     if TheMessage = WM_VSCROLL then begin

      {Get the min and max range of the horizontal scroll box}

      GetScrollRange(WindowHandle, SB_HORZ, TheRangeMin, TheRangeMax);

      {Get the vertical scroll box position}

      TheRange := GetScrollPos(WindowHandle, SB_VERT);

      {Make sure we wont exceed the range}

      if TheRange < TheRangeMin then TheRange := TheRangeMin else if TheRange > TheRangeMax then TheRange := TheRangeMax;

      {Set the horizontal scroll bar}

      SetScrollPos(WindowHandle, SB_HORZ, TheRange, true);

     end;

     if TheMessage = WM_HSCROLL then begin

      {Get the min and max range of the horizontal scroll box}

      GetScrollRange(WindowHandle, SB_VERT, heRangeMin, TheRangeMax);

      {Get the horizontal scroll box position}

      TheRange := GetScrollPos(WindowHandle, SB_HORZ);

      {Make sure we wont exceed the range}

      if TheRange < TheRangeMin then TheRange := TheRangeMin else if TheRange > TheRangeMax then TheRange := TheRangeMax;

       {Set the vertical scroll bar}

       SetScrollPos(WindowHandle, SB_VERT, TheRange, true);

      end;

     {Call the old Window procedure to allow processing of the message.}

     NewWindowProc := CallWindowProc(OldWindowProc, WindowHandle, TheMessage, ParamW, ParamL);

    end;


    procedure TForm1.FormCreate(Sender: TObject);

    begin

     {Set the new window procedure for the control and remember the old window procedure.}

     OldWindowProc := Pointer(SetWindowLong(ScrollBox1.Handle, GWL_WNDPROC, LongInt(@NewWindowProc)));

    end;


    procedure TForm1.FormDestroy(Sender: TObject);

    begin

     {Set the window procedure back to the old window procedure.}

     SetWindowLong(ScrollBox1.Handle, GWL_WNDPROC, LongInt(OldWindowProc));

    end;


    Вопрос:

    Как сделать прямоугольник для выделения части картинки для редактирования?

    Ответ:

    Самый простой способ — воспользоваться функцией Windows API DrawFocusRect. Функция DrawFocusRect использует операцию XOR при рисовании — таким образом вывод прямоугольника дважды с одними и теми же координатами стирает прямоугольник, и прямоугольник всегда будет виден, на фоне какого бы цвета он не выводился.

    Пример:

    type TForm1 = class(TForm)

     procedure FormMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

     procedure FormMouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

     procedure FormMouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    private

     {Private declarations}

     Capturing : bool;

     Captured : bool;

     StartPlace : TPoint;

     EndPlace : TPoint;

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation

    {$R *.DFM}


    function MakeRect(Pt1 : TPoint; Pt2 : TPoint) : TRect;

    begin

     if pt1.x < pt2.x then begin

      Result.Left := pt1.x;

      Result.Right := pt2.x;

     end else begin

      Result.Left := pt2.x;

      Result.Right := pt1.x;

     end;

     if pt1.y < pt2.y then begin

      Result.Top := pt1.y;

      Result.Bottom := pt2.y;

     end else begin

      Result.Top := pt2.y;

      Result.Bottom := pt1.y;

     end;

    end;


    procedure TForm1.FormMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    begin

     if Captured then DrawFocusRect(Form1.Canvas.Handle,MakeRect(StartPlace,EndPlace));

     StartPlace.x := X;

     StartPlace.y := Y;

     EndPlace.x := X;

     EndPlace.y := Y;

     DrawFocusRect(Form1.Canvas.Handle,MakeRect(StartPlace,EndPlace));

     Capturing := true;

     Captured := true;

     end;


    procedure TForm1.FormMouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);

    begin

     if Capturing then begin

      DrawFocusRect(Form1.Canvas.Handle,MakeRect(StartPlace,EndPlace));

      EndPlace.x := X;

      EndPlace.y := Y;

      DrawFocusRect(Form1.Canvas.Handle,MakeRect(StartPlace,EndPlace));

     end;

    end;


    procedure TForm1.FormMouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);

    begin

     Capturing := false;

    end;


    Вопрос:

    Можно ли использовать иконку как картинку на кнопке TSpeedButton?

    Ответ:

    Можно.

    См. пример.

    Пример:

    uses ShellApi;


    procedure TForm1.FormShow(Sender: TObject);

    var Icon: TIcon;

    begin

     Icon := TIcon.Create;

     Icon.Handle := ExtractIcon(0,'C:\WINDOWS\NOTEPAD.EXE',1);

     SpeedButton1.Glyph.Width := Icon.Width;

     SpeedButton1.Glyph.Height := Icon.Height;

     SpeedButton1.Glyph.Canvas.Draw(0, 0, Icon);

     Icon.Free;

    end;


    Вопрос:

    Как поместить прозрачную фоновую каринку на компонент CoolBar?

    Ответ:

    procedure TForm1.Button1Click(Sender: TObject);

    var

     Bm1 : TBitmap;

     Bm2 : TBitmap;

    begin

     Bm1 := TBitmap.Create;

     Bm2 := TBitmap.Create;

     Bm1.LoadFromFile('c:\download\test.bmp');

     Bm2.Width := Bm1.Width;

     Bm2.Height := Bm1.Height;

     bm2.Canvas.Brush.Color := CoolBar1.Color;

     bm2.Canvas.BrushCopy(Rect(0, 0, bm2.Width, bm2.Height), Bm1, Rect(0, 0, Bm1.width, Bm1.Height), ClWhite);

     bm1.Free;

     CoolBar1.Bitmap.Assign(bm2);

     bm2.Free;

    end;


    Вопрос:

    Ползунок компонента TScrollBar все время мигает. Как это отключить?

    Ответ:

    Установите свойтсво ScrollBar.TabStop в False.


    Вопрос:

    Как программно перевести DBgrid в реим редактирования и установить курсор в окошке редактирования в требуемую позицию?

    Ответ:

    Переведите таблицу в режим редактирования, затем получите дескриптор (handle) окна редактирования и перешлите ей сообщение EM_SETSEL. В качестве параметров вы должны переслать начальную позицию курсора, и конечную позицию, определяющую конец выделения текста цветом. В приведенном примере курсор помещается во вторую позицию, текст внутри ячейки не выделяется.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    var h : THandle;

    begin

     Application.ProcessMessages;

     DbGrid1.SetFocus;

     DbGrid1.EditorMode := true;

     Application.ProcessMessages;

     h:= Windows.GetFocus;

     SendMessage(h, EM_SETSEL, 2, 2);

    end;


    Вопрос:

    Как поместить курсор в определенную позицию edit'а и подобных ему элементов управления?

    Ответ:

    Можно использовать методы Delphi SelStart() и SelectLength().

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Edit1.SetFocus;

     {переводим курсор во вторую позицию}

     Edit1.SelStart := 2;

     {не выделяем никакого текста}

     Edit1.SelLength := 0;

    end;


    Вопрос:

    Как среагировать на минимизацию-максимизацию формы перед тем как произойдет изменение размера формы?

    Ответ:

    В примере перехватывается сообщение WM_SYSCOMMAND. Если это сообщение говорит о минимизации или максимизации формы — пищит динамик.

    Пример:

    type TForm1 = class(TForm)

    private

     {Private declarations}

     procedure WMSysCommand(var Msg: TWMSysCommand); message WM_SYSCOMMAND;

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation

    {$R *.DFM}


    procedure TForm1.WMSysCommand;

    begin

     if (Msg.CmdType = SC_MINIMIZE) or (Msg.CmdType = SC_MAXIMIZE) then MessageBeep(0)

     else inherited;

    end;


    Вопрос:

    Можно ли сделать так — одна форма показывает другую и остается позади нее, но фокус ввода не переходит к новой форме, а остается у старой?

    Ответ:

    В примере показывается не автосоздаваемая (non auto-created) форма, но фокус ввода ей не передается.

    Пример:

    uses Unit2;


    procedure TForm1.Button1Click(Sender: TObject);

    begin

     Form2 := TForm2.Create(Application);

     Form2.Visible := FALSE;

     ShowWindow(Form2.Handle, SW_SHOWNA);

    end;


    Вопрос:

    На некоторых laptop компьютерах может не быть флоппи дисковода. Можно ли удалять из списка TDriveComboBox диски которые отключены?

    Ответ:

    В примере TDriveComboBox не показывает дисководы, которые не готовы. (not ready). Учтите что на многих компьютерах будет ощутимая задержка при поверке plug&play флоппи дисковода.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    var

     i : integer;

     OldErrorMode : Word;

     OldDirectory : string;

    begin

     OldErrorMode := SetErrorMode(SEM_NOOPENFILEERRORBOX);

     GetDir(0, OldDirectory);

     i := 0;

     while i <= DriveComboBox1.Items.Count - 1 do begin

      {$I-}

      ChDir(DriveComboBox1.Items[i][1] + ':\');

      {$I+}

      if IoResult <> 0 then DriveComboBox1.Items.Delete(i)

      else inc(i);

     end;

     ChDir(OldDirectory);

     SetErrorMode(OldErrorMode);

    end;


    Вопрос:

    Как сообщить всем формам моего приложения (в том числе и не видимым в данный момент) об изменении каких-то глобальных значений?

    Ответ:

    Один из способов — создать пользовательское сообщение и использовать метод preform чтобы разослать его всем формам из массива Screen.Forms.

    Пример:

    {Code for Unit1}

    const UM_MyGlobalMessage = WM_USER + 1;

    type TForm1 = class(TForm)

     Label1: TLabel;

     Button1: TButton;

     procedure FormShow(Sender: TObject);

     procedure Button1Click(Sender: TObject);

    private

     {Private declarations}

     procedure UMMyGlobalMessage(var AMessage: TMessage); message UM_MyGlobalMessage;

    public

     {Public declarations}

    end;


    var Form1: TForm1;


    implementation

    {$R *.DFM}


    uses Unit2;


    procedure TForm1.FormShow(Sender: TObject);

    begin

     Form2.Show;

    end;


    procedure TForm1.UMMyGlobalMessage(var AMessage: TMessage);

    begin

     Label1.Left := AMessage.WParam;

     Label1.Top := AMessage.LParam;

     Form1.Caption := 'Got It!';

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    var f: integer;

    begin

     for f := 0 to Screen.FormCount - 1 do Screen.Forms[f].Perform(UM_MyGlobalMessage, 42, 42);

    end;


    {Code for Unit2}

    const UM_MyGlobalMessage = WM_USER + 1;

    type TForm2 = class(TForm)

     Label1: TLabel;

    private

     {Private declarations}

     procedure UMMyGlobalMessage(var AMessage: TMessage); message UM_MyGlobalMessage;

    public

     {Public declarations}

    end;


    var Form2: TForm2;


    implementation

    {$R *.DFM}


    procedure TForm2.UMMyGlobalMessage(var AMessage: TMessage);

    begin

     Label1.Left := AMessage.WParam;

     Label1.Top := AMessage.LParam;

     Form2.Caption := 'Got It!';

    end;


    Вопрос:

    Как обновить список дисков компонента TDriveComboBox, учитывая, что могут быть подключены/отключены сетевые диски и произведена "горячая замена" plug&play дисков?

    Ответ:

    Следующий пример вызывает защищенный (protected) метод класса TDriveComboBox BuildList() для регенерации списка дисков. (использовая так наз. "class cracer")

    Пример:

    type TNewDriveComboBox = class(TDriveComboBox) //это наш "class cracer"

    end;


    procedure TForm1.Button1Click(Sender: TObject);

    var Drive : char;

    begin

     Drive := DriveComboBox1.Drive;

     TNewDriveComboBox(DriveComboBox1).BuildList; //вызываем защищенный метод родительского класса

     DriveComboBox1.Drive := Drive;

    end;


    Вопрос:

    Как программно заставить выпасть меню?

    Ответ:

    В примере показано как показать меню и выбрать в нем какой-то пункт, эмулируя нажатие "быстрой клавиши" пункта меню. Если у Вашего пункта меню нет "быстрой клавиши" Вы можете посылать комбинации VK_MENU, VK_LEFT, VK_DOWN, и VK_RETURN, чтобы программно "путешествовать" по меню.

    Пример:

    procedure TForm1.Button1Click(Sender: TObject);

    begin

     //Allow button to finish painting in response to the click

     Application.ProcessMessages;

     {Alt Key Down}

     keybd_Event(VK_MENU, 0, 0, 0);

     {F Key Down - Drops the menu down}

     keybd_Event(ord('F'), 0, 0, 0);

     {F Key Up}

     keybd_Event(ord('F'), 0, KEYEVENTF_KEYUP, 0);

     {Alt Key Up}

     keybd_Event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);

     {F Key Down}

     keybd_Event(ord('S'), 0, 0, 0);

     {F Key Up}

     keybd_Event(ord('S'), 0, KEYEVENTF_KEYUP, 0);

    end;


    Вопрос:

    Как сделать клавишу-акселератор (keyboard shortcut) компоненту, у которого нет заголовка?

    Ответ:

    Возможный вариант — присвоить ссылку на этот компонент свойству FocusControl TLabel'а. В примере используется невидимый Label для создания "быстрой" клавиши (Alt+M) компонента Memo. Чтобы использовать пример, разместите на форме компонет TMemo, Label и несколько других компонентов, которые могут принимать фокус ввода. Запустите программу, перевидите фокус ввода куда-нибудь вне Memo и нажмите Alt+M — фокус ввода вернется в Memo.

    Пример:

    procedure TForm1.FormCreate(Sender: TObject);

    begin

     Label1.Visible := false;

     Label1.Caption := '&M';

     Label1.FocusControl := Memo1;

    end;


     Вопрос:

    Можно ли как-то уменьшить мерцание при перерисовке компонента?

    Ответ:

    Если добавить флаг csOpaque (непрозрачный) к свойству ControlStyle компонента — то фон компонента перерисовываться не будет.

    Пример:

    constructor TMyControl.Create;

    begin

     inherited;

     ControlStyle := ControlStyle + [csOpaque];

    end;


    Вопрос:

    Как запретить изменение размера моего компонента в design-time?

    Ответ:

    Поместите в конструктор компонента код, устанавливающий размеры по умолчанию. Переопределите метод SetBounds и проверяйте в нем "componentstate". Если компонет находится режиме "design-time" (csDesigning in ComponentState) просто передавайте значения ширины и высоты (width и heights) компонента по умолчанию (в нашем примере 50) методу класса-предка.

    Пример:

    procedure TVu.SetBounds(ALeft : integer; ATop : integer; AWidth : integer; AHeight : integer);

    begin

     if csdesigning in componentstate then begin

      AWidth := 50;

      AHeight := 50;

      inherited; //вызываем унаследованный от предка метод

     end;

    end;


    Вопрос:

    Можно ли уменьшить потребляемые компонентами TNotebook и TTabbedNotebook ресурсы?

    Ответ:

    Да. Можно уничтожать обьекты, расположенные не на текущей странице TNotebook или TTabbedNotebook. В примере вызывается защищенный (Protected) метод путем создания так называемый "class cracer'ов".

    type TMyTabbedNotebook = class(TTabbedNotebook); //это наш "class cracer"

    type TMyNotebook = class(TNotebook);


    procedure TForm1.TabbedNotebook1Change(Sender: TObject; NewTab: Integer; var AllowChange: Boolean);

    begin

     with TabbedNotebook1 do //вызываем защищенный метод родительского класса

      TMyTabbedNotebook(TWinControl(Pages.Objects[PageIndex])).DestroyHandle;

    end;


    procedure TForm1.TabSet1Change(Sender: TObject; NewTab: Integer; var AllowChange: Boolean);

    begin

     with Notebook1 do //вызываем защищенный метод родительского класса

      TMyNotebook(TWinControl(Pages.Objects[PageIndex])).DestroyHandle;

     NoteBook1.PageIndex := NewTab;

     AllowChange := true

    end;


    Вопрос:

    Функция keybd_event() принимает значения до 244 — как мне отправить нажатие клавиши с кодом #255 в элемент управления Windows?

    Ответ:

    Это может понадобится для иностранных языков или для специальных символов. (например, в русских шрифтах символ с кодом #255 — я прописное). Приведенный в примере метод не стоит использовать в случае, если символ может быть передан обычным способом (функцией keybd_event()).

    procedure TForm1.Button1Click(Sender: TObject);

    var KeyData : packed record

     RepeatCount : word;

     ScanCode : byte;

     Bits : byte;

    end;

    begin

     {Let the button repaint}

     Application.ProcessMessages;

     {Set the focus to the window}

     Edit1.SetFocus;

     {Send a right so the char is added to the end of the line}

     // SimulateKeyStroke(VK_RIGHT, 0);

     keybd_event(VK_RIGHT, 0,0,0);

     {Let the app get the message}

     Application.ProcessMessages;

     FillChar(KeyData, sizeof(KeyData), #0);

     KeyData.ScanCode := 255;

     KeyData.RepeatCount := 1;

     SendMessage(Edit1.Handle, WM_KEYDOWN, 255,LongInt(KeyData));

     KeyData.Bits := KeyData.Bits or (1 shl 30);

     KeyData.Bits := KeyData.Bits or (1 shl 31);

     SendMessage(Edit1.Handle, WM_KEYUP, 255, LongInt(KeyData));

     KeyData.Bits := KeyData.Bits and not (1 shl 30);

     KeyData.Bits := KeyData.Bits and not (1 shl 31);

     SendMessage(Edit1.Handle, WM_CHAR, 255, LongInt(KeyData));

     Application.ProcessMessages;

    end;


    Вопрос:

    Некоторые компоненты не меняют курсор мыши до тех пор пока пользователь не сдвинет мышь. Как эмулировать движение мыши?

    Ответ:

    В примере мышка слегка "подталкивается" без участия пользователя.

    procedure TForm1.Button1Click(Sender: TObject);

    var pt : TPoint;

    begin

     Application.ProcessMessages;

     Screen.Cursor := CrHourglass;

     GetCursorPos(pt);

     SetCursorPos(pt.x + 1, pt.y + 1);

     Application.ProcessMessages;

     SetCursorPos(pt.x - 1, pt.y - 1);

    end;


    Вопрос:

    Как зарегистрировать расширение файла за своим приложением и контекстное меню, связанное с этим типом?

    Ответ:

    Пример регистрирует расширение файла(.myext) — файлы этого типа будут открываться приложением MyApp.Exe. Также регнстрируется одно действие (action) по умолчанию для файлов этого типа и два дополнительных пункта контекстного меню, связанного с этим типом файлов. Возможно, потребуется перезайти в систему чтобы изменения вступили в силу.

    Пример:

    uses Registry;


    procedure TForm1.Button1Click(Sender: TObject);

    var R : TRegIniFile;

    begin

     R := TRegIniFile.Create('');

     with R do begin

      RootKey := HKEY_CLASSES_ROOT;

      WriteString('.myext','','MyExt');

      WriteString('MyExt','','Some description of MyExt files');

      WriteString('MyExt\DefaultIcon','','C:\MyApp.Exe,0');

      WriteString('MyExt\Shell','','This_Is_Our_Default_Action');

      WriteString('MyExt\Shell\First_Action', '','This is our first action');

      WriteString('MyExt\Shell\First_Action\command','', 'C:\MyApp.Exe /LotsOfParamaters %1');

      WriteString('MyExt\Shell\This_Is_Our_Default_Action','', 'This is our default action');

      WriteString('MyExt\Shell\This_Is_Our_Default_Action\command', '','C:\MyApp.Exe %1');

      WriteString('MyExt\Shell\Second_Action', '','This is our second action');

      WriteString('MyExt\Shell\Second_Action\command', '','C:\MyApp.Exe /TonsOfParameters %1');

      Free;

     end;

    end;


    Примечания:



    1

    использовались только версии ODBC 2.0 и Access 2.0.



    2

    Local InterBase — однопользовательский SQL сервер базы данных. Версия, включенная в Delphi Desktop, предназначается для использования разработчиками, которые хотят разрабатывать SQL приложения (для последующего переноса их в среду клиент/сервер) без покупки собственной (дорогой) платформы сервера. Однако, Delphi Desktop не включает права распространения на Local InterBase. Если вы хотите распространять однопользовательское приложение, которое его использует, вы должны заплатить дополнительно за deployment kit.  



    3

     Версия ReportSmith из Delphi Desktop специально обнаруживает и исключает из списка возможных соединений любой ODBC драйвер к удаленному серверу данных. Да, вы не ослышались. Даже если вы купили ODBC драйвер третьей фирмы, и даже если вы можете прекрасно общаться с этим драйвером из Delphi, ReportSmith еще не будет с ним работать. Если вы хотите сделать это без покупки Delphi Client/Server, вы может купить ReportSmith/SQL отдельно за $300 (и тем не менее приобретете ReportSmith версии 2.0, а не 2.5, как в поставке Delphi Client/Server).  







     


    Главная | В избранное | Наш E-MAIL | Добавить материал | Нашёл ошибку | Наверх