Программируем на Visual Basic'e:

Часть 1/Часть 2/Часть 3/Часть 4/Часть 5/Часть 6/Часть 7/Часть 8/ Часть 9/Часть 10/Часть 11/Часть 12 /Часть 13/Часть 14 /Часть 15 /Часть 16 /Часть 17 /Часть 18

ШАГ 161

Делаем генератор ключа

Хотя этот шаг не стоило писать, т.к. это очень примитивная программка. Эта програмка будет по с/н(который пришлет покупатель программы) делать ключ:
На форму кинь кнопку(Caption = "Crack"), 2 текст. поля(в ряд), 2 метки(у первой Caption = "Введите Серийный номер", она находится напротив первого текст. поля, у второй: Caption = "Ключ", напротив второго текст. поля). Вот что у тебя должно получиться:



Теперь код:

Private Sub Command1_Click()
Text2.Text = Fix(Val(Text1.Text) * 3 + 333 / 2) ' По С/Н узнаем ключ
End Sub

Исходник ищи Здесь.

ШАГ 162

Оператор Goto

Этот оператор служит для пропуска определенной части кода.
Синтаксис:
Goto метка:
Метка:
Весь код от Goto метка: до метка: пропускается.
Например, сейчас сделаем программу которая будет перескакивать оператор Exit Sub. На форму кинь кнопку.
Вот Код:

Dim x As Long

Private Sub Command1_Click()
x = x + 1 ' При каждом клике по кнопке к перем. x прибавляем 1
If x = 2 Then GoTo qw: ' Если x = 2 то с помощью оператора Goto пропускаем весь код до qw:
Exit Sub ' Выходим из процедуры
qw: ' Метка, до которой пропускается весь код от оператора Goto qw:
MsgBox "Оператор Goto" ' Выдаем сообщение
End Sub Исходник ищи Здесь.

ШАГ 163

Файлы ресурсов

В файлах ресурсов можно хранить картинки, текст, аудио... Для чего это нужно?:
Например, ты делаешь программу в которой очень много рисунков. Можно все рисунки положить в отдельную папку, но это будет не прикольно(т.к. любой смертный сможет удалить/изменить эти рисунки). А если положить все рисунки в файл ресурсов, то все рисунки откомпилируются в 1 exe файл, и никто их не изменит. Вот так :-)
Начнем`c.
Открой VB, Standart EXE. Нажми Дополнения(Add ins) >> Менеджер дополнений(Add ins Manager). Открывается окошко, в нем найди строку: VB 6 Resource Editor, и в нижнем правом углу поставь две галки на: Startup/Loaded и Startup/Unloaded(Загрузить/выгрузить и Загрузить при старте). На верхней панели у тебя должен появится такой значок:
Нажми на него:

ШАГ 164

Загрузка из файлов ресурсов

Для загрузки из файлов ресурсов есть 3 функции:
LoadResData
LoadResPicture
LoadResString

LoadResString - загружает строку из файла. Пример:
Private Sub Command1_Click()
Text1.Text = LoadResString(101) ' Загружаем текст из файла ресурсов, с id номером 101
End Sub
LoadResPicture - Функция загружает изображение, значок или курсор.
Синтаксис: LoadResPicture(id, restype)
id - номер рисунка(например, 102)
restype - значение этого параметра указывают на тип файла(0 - изображение, 1 - значок, 2 - курсор)
С LoadResData я так и не разобрался, если ты знаешь как этой функцией пользоваться, то напиши мне. ОК?

ШАГ 165

Работа с Image. Масштабирование.

Сейчас мы сделаем программу, которая будет уменьшать/увеличивать рисунок. На форму кинь 2 кнопки(1ая - Caption = "+", у второй "-"), и объект Image(свойство stretch = True (в этом свойстве все и заключается)).

Private Sub Command1_Click()' Увеличиваем
Image1.Width = Image1.Width + 100
Image1.Height = Image1.Height + 100
End Sub

Private Sub Command2_Click()' Уменьшаем масштаб
Image1.Width = Image1.Width - 100
Image1.Height = Image1.Height - 100
End Sub

Исходник - Здесь.

ШАГ 166

Работа с файлами произвольного доступа

Файл с произвольного доступа обладает заданной структурой и состоит из записей(также как и БД). Каждая запись имеет свой номер в файле. Доступ к данным осуществляется по этому номеру. Т.е. мы можем использовать обычный текстовой файл как базу данных.
Открытие файла производится только в режиме Random. Для записи и чтения существуют операторы Put и Get
Напишем простенькую программу, которая будет записывать и читать из текст. файла записи. На форму кинь одно текстовое поле, и 2 кнопки(Command1 - Caption = "Запись", Command2 - Caption = "Чтение"). Код

' Объявляем переменные
Dim PutNomer As Long, GetNomer As Long, Gets As String

Private Sub Command1_Click()' При нажатии на кнопку "Запись":
PutNomer = PutNomer + 1' переменная PutNomer - это номер записи, при каждом нажатии на кнопку, на 1 больше
Open "c:\1.txt" For Random As #1' Открываем файл 1.txt, если его нет, то он создастся, и еше, я написал #1 - это номер свободного канала, вместо него можно использовать функции FreeFile
Put #1, PutNomer, Text1.Text' Записываем текст, который в текст. поле
Close #1' Закрываем файл:
End Sub

Private Sub Command2_Click()' При нажатии на кнопку "Чтение":
GetNomer = GetNomer + 1' Переменная GetNomer становится на 1 больше
Open "c:\1.txt" For Random As #1
Get #1, GetNomer, Gets' Читаем текст в переменную Gets, по номеру записи
Close #1
Text1.Text = Gets' То, что прочитали выводим в текст. поле
End Sub

Ну вот, вроде бы и все, этот пример содержит кучу багов, но это не важно, главное научится обращаться с этими функциями. Исходник.

ШАГ 167

Работа с меню. Всовывание картинок в меню.

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



Начнем`с..
Для начала найди картинки(размер приблизатильно 14x14, можно побольше, можно поменьше), можешь взять мои:

Теперь, на форму кинь 4 Image.
В Image1.Picture = Картинка 1
В Image2.Picture = Картинка 2
В Image3.Picture = Картинка 3
В Image4.Picture = Картинка 4

Теперь создай меню, из следующих разделов:

Файд
....Создать
....Открыть
....Сохранить
....Сохранить как
....-
....Параметры страницы
....Печать
....-
....Выход

Имена разделам сам придумай, от них все равно ни чего не зависит.
Немного поясню, вот этот раздел: "....-" нужен для рисования линии в меню
Теперь перейдем к коду:

' Объявляем API
Private Declare Function GetMenu Lib "user32" (ByVal hWnd As Long) As Long
Private Declare Function GetSubMenu Lib "user32" (ByVal hMenu As Long, ByVal nPos As Long) As Long
Private Declare Function GetMenuItemID Lib "user32" (ByVal hMenu As Long, ByVal nPos As Long) As Long
Private Declare Function SetMenuItemBitmaps Lib "user32" (ByVal hMenu As Long, ByVal nPosition As Long, ByVal wFlags As Long, ByVal hBitmapUnchecked As Long, ByVal hBitmapChecked As Long) As Long
Private Const MF_BITMAP = &H4&

Private Sub Form_Load()
Dim M As Long, B As Long, H As Long, G As Long' Объявляем переменные

M = GetMenu(Form1.hWnd)
B = GetSubMenu(M, 0)

' H = GetMenuItemID(B, Здесь надо указать позицию меню, напротив которого будет картинка находиться)
H = GetMenuItemID(B, 2)
G = SetMenuItemBitmaps(M, H, MF_BITMAP, Image3.Picture, Image3.Picture)' Здесь надо указать имя image, в котором будет находиться картинка

H = GetMenuItemID(B, 3)
G = SetMenuItemBitmaps(M, H, MF_BITMAP, Image2.Picture, Image2.Picture)

H = GetMenuItemID(B, 5)
G = SetMenuItemBitmaps(M, H, MF_BITMAP, Image1.Picture, Image1.Picture)

H = GetMenuItemID(B, 6)
G = SetMenuItemBitmaps(M, H, MF_BITMAP, Image4.Picture, Image4.Picture)
End Sub

Исходник Здесь.

ШАГ 168

Правильная компиляция

Меня часто спрашивают, чем Compile to P-code лучше Compile to Native Code, вот в этом шаге об этом и пойдет речь(ну и о другом).
Для начала войди в Проект>>Project Свойства.

Вкладка General
1. Тип проекта - здесь надо указать тип проекта, т.е. Standart EXE, ActiveX, Active DLL.. Если ты создаешь обычный exe проект, то оставь Standart exe.
2. Стандартный объект запуска - здесь надо указать какая форма будет запускаться первой(при запуске программы). Если указать Sub Main, то надо создать процедуру Sub Main(обычно в модуле):

Public Sub Main()
Код
End Sub

3. Имя проекта - здесь надо указать имя проекта.
4. Имя - здесь надо путь к справке(об этом потом)

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

Вкладка Compile
В этой вкладке надо выбрать тип компиляции.
1. Компилировать с Р-кодом - если компилировать проект в Р код, то программа будет запускаться только с установленным Visual Basic`om
2. Компилировать с Native кодом - это обычная компиляция в машинный код, у нее есть дополнительные параметры:
2.1. Оптимизация под быстрый код - Программа работает быстрее, но занимает больше места
2.2. Оптимизация под малый код - Программа работает медленнее, но занимает меньше места
2.3. Оптимизация под малый код - Программа работает медленнее, но занимает меньше места
2.4. Не оптимизировать - Без оптимизайции
2.5. Favor Pentium Pro - Программа работает быстрее на Pentium Pro
2.6. Create Symbolic Debug Info - Создается отладочный файл

ШАГ 169

API функция - "GetAsyncKeyState"

Функция GetAsyncKeyState нужна для отслжевания нажатых клавиш, если форма не активна. Вот маленький пример использования функции. На форму кинь таймер(Interval = 1), и впиши код:

' Объявляем API
Private Declare Function GetAsyncKeyState Lib "user32" (ByVal vKey As Long) As Integer

Private Sub Timer1_Timer()
If GetAsyncKeyState(32) = -32768 Then MsgBox "Нажат Пробел"' Функция GetAsyncKeyState(ASCII код клавиши) возвращает -32768, если нажата клавиша, ASCII код которой находится в скобках, и возвращает 0, если не нажата
If GetAsyncKeyState(vbKeyF1) = -32768 Then MsgBox "Нажат F1"
End Sub

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

ШАГ 170

Регистрация новых типов файлов(расширений)

Для регистрации новых расширений нужно воспользоваться реестром. Сейчас мы сделаем программу, которая зарегистрирует новое расширение, и даст ему иконку. На форму кинь кнопку, и создай модуль. Код модуля(взят из шага 74, работа с реестром. Модуль нужен для работы с реестром.):
Код Модуля:
'--------------------------------------------------------------------------------------------------

Public Const REG_SZ As Long = 1
Public Const REG_DWORD As Long = 4
Public Const HKEY_LOCAL_MACHINE = &H80000002
Public Const HKEY_CLASSES_ROOT = &H80000000
Public Const HKEY_CURRENT_USER = &H80000001
Public Const HKEY_USERS = &H80000003


Public Const ERROR_NONE = 0
Public Const ERROR_BADDB = 1
Public Const ERROR_BADKEY = 2
Public Const ERROR_CANTOPEN = 3
Public Const ERROR_CANTREAD = 4
Public Const ERROR_CANTWRITE = 5
Public Const ERROR_OUTOFMEMORY = 6
Public Const ERROR_INVALID_PARAMETER = 7
Public Const ERROR_ACCESS_DENIED = 8
Public Const ERROR_INVALID_PARAMETERS = 87
Public Const ERROR_NO_MORE_ITEMS = 259
Public Const KEY_ALL_ACCESS = &H3F
Public Const REG_OPTION_NON_VOLATILE = 0

Declare Function RegCloseKey Lib "advapi32.dll" (ByVal hKey As Long) As Long
Declare Function RegCreateKeyEx Lib "advapi32.dll" Alias "RegCreateKeyExA" (ByVal hKey As Long, ByVal lpSubKey As String, ByVal Reserved As Long, ByVal lpClass As String, ByVal dwOptions As Long, ByVal samDesired As Long, ByVal lpSecurityAttributes As Long, phkResult As Long, lpdwDisposition As Long) As Long
Declare Function RegOpenKeyEx Lib "advapi32.dll" Alias "RegOpenKeyExA" (ByVal hKey As Long, ByVal lpSubKey As String, ByVal ulOptions As Long, ByVal samDesired As Long, phkResult As Long) As Long
Declare Function RegQueryValueExString Lib "advapi32.dll" Alias "RegQueryValueExA" (ByVal hKey As Long, ByVal lpValueName As String, ByVal lpReserved As Long, lpType As Long, ByVal lpData As String, lpcbData As Long) As Long
Declare Function RegQueryValueExLong Lib "advapi32.dll" Alias "RegQueryValueExA" (ByVal hKey As Long, ByVal lpValueName As String, ByVal lpReserved As Long, lpType As Long, lpData As Long, lpcbData As Long) As Long
Declare Function RegQueryValueExNULL Lib "advapi32.dll" Alias "RegQueryValueExA" (ByVal hKey As Long, ByVal lpValueName As String, ByVal lpReserved As Long, lpType As Long, ByVal lpData As Long, lpcbData As Long) As Long
Declare Function RegSetValueExString Lib "advapi32.dll" Alias "RegSetValueExA" (ByVal hKey As Long, ByVal lpValueName As String, ByVal Reserved As Long, ByVal dwType As Long, ByVal lpValue As String, ByVal cbData As Long) As Long
Declare Function RegSetValueExLong Lib "advapi32.dll" Alias "RegSetValueExA" (ByVal hKey As Long, ByVal lpValueName As String, ByVal Reserved As Long, ByVal dwType As Long, lpValue As Long, ByVal cbData As Long) As Long
Declare Function RegDeleteKey& Lib "advapi32.dll" Alias "RegDeleteKeyA" (ByVal hKey As Long, ByVal lpSubKey As String)
Declare Function RegDeleteValue& Lib "advapi32.dll" Alias "RegDeleteValueA" (ByVal hKey As Long, ByVal lpValueName As String) 'Создание нового ключа
Public Function CreateNewKey(lPredefinedKey As Long, sNewKeyName As String)
Dim hNewKey As Long
Dim lRetVal As Long
lRetVal = RegCreateKeyEx(lPredefinedKey, sNewKeyName, 0&, vbNullString, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, 0&, hNewKey, lRetVal)
RegCloseKey (hNewKey)
End Function

'Запись данных в ключ
Public Function SetKeyValue(lPredefinedKey As Long, sKeyName As String, sValueName As String, vValueSetting As Variant, lValueType As Long)
Dim lRetVal As Long
Dim hKey As Long


lRetVal = RegOpenKeyEx(lPredefinedKey, sKeyName, 0, KEY_ALL_ACCESS, hKey)
lRetVal = SetValueEx(hKey, sValueName, lValueType, vValueSetting)

RegCloseKey (hKey)
End Function

Public Function SetValueEx(ByVal hKey As Long, sValueName As String, lType As Long, vValue As Variant) As Long

Dim lValue As Long
Dim sValue As String

Select Case lType
Case REG_SZ
sValue = vValue
SetValueEx = RegSetValueExString(hKey, sValueName, 0&, lType, sValue, Len(sValue))
Case REG_DWORD
lValue = vValue
SetValueEx = RegSetValueExLong(hKey, sValueName, 0&, lType, lValue, 4)
End Select

End Function

'Возвращает значения записанные в ключе(т.е. чтение)
Public Function QueryValue(lPredefinedKey As Long, sKeyName As String, sValueName As String)

Dim lRetVal As Long
Dim hKey As Long
Dim vValue As Variant
lRetVal = RegOpenKeyEx(lPredefinedKey, sKeyName, 0, KEY_ALL_ACCESS, hKey)
lRetVal = QueryValueEx(hKey, sValueName, vValue)
QueryValue = vValue
RegCloseKey (hKey)

End Function

Function QueryValueEx(ByVal lhKey As Long, ByVal szValueName As String, vValue As Variant) As Long

Dim cch As Long
Dim lrc As Long
Dim lType As Long
Dim lValue As Long
Dim sValue As String

On Error GoTo QueryValueExError

'Определение размера и типа считываемых данных
lrc = RegQueryValueExNULL(lhKey, szValueName, 0&, lType, 0&, cch)
If lrc <> ERROR_NONE Then MsgBox "Данных (ключа) не существует!", vbExclamation, Form1.Caption
Select Case lType
'Для символьных
Case REG_SZ:
sValue = String(cch, 0)
lrc = RegQueryValueExString(lhKey, szValueName, 0&, lType, sValue, cch)
If lrc = ERROR_NONE Then
vValue = Left$(sValue, cch)
Else
vValue = Empty
End If
'Для числовых
Case REG_DWORD:
lrc = RegQueryValueExLong(lhKey, szValueName, 0&, lType, lValue, cch)
If lrc = ERROR_NONE Then vValue = lValue
'Для остальных не поддержанных типов данных
Case Else
lrc = -1
End Select
QueryValueExExit:
QueryValueEx = lrc
Exit Function
QueryValueExError:
Resume QueryValueExExit
End Function

'Удаление значений ключа
Public Function DeleteValue(lPredefinedKey As Long, sKeyName As String, sValueName As String)
Dim lRetVal As Long
Dim hKey As Long
lRetVal = RegOpenKeyEx(lPredefinedKey, sKeyName, 0, KEY_ALL_ACCESS, hKey)
lRetVal = RegDeleteValue(hKey, sValueName)
RegCloseKey (hKey)
End Function

'Удаление ключа
Public Function DeleteKey(lPredefinedKey As Long, sKeyName As String)
Dim lRetVal As Long
lRetVal = RegDeleteKey(lPredefinedKey, sKeyName)
End Function

'-------------------------------------------------------------------------------------------------------

Теперь код формы. Скажу последовательность действий:
1. Надо создать ключ в реестре с разделе HKEY_CLASSES_ROOT с нужным расширением(например ".cdk").
2. Значение стокового параметра "(По умолчанию)" в созданном ключе, надо изменить на имя своей программы(например "cdkfile"(вместо ""))
3. Надо создать еще один ключ, с тем именем, который ты внес в значение строкого параметра "По умолчанию" предыдущем ключе(пред. ключ ".cdk", значение "cdkfile"). В этом ключе мы и будем создавать ключи, отвечающие за какие - либо действия(например открытие файла, или присвоение иконки, или печать..)
4. Значение срокового параметра "По умолчанию" ключа("cdkfile") можно изменить на любой текст. И этот текст будет отображаться в свойствах файла, в разделе "Общие>Тип Файла", если стоит XP, то под файлом.
5. Для прикрепления иконки к созданному расширению, нодо создать в ключе "cdkfile" подключ "DefaultIcon", и значение строкового параметра "По умолчанию" надо изменить на путь к иконке(причем путь может быть не только в *.ico файлам, но и к *.exe файлам. И тогда иконка возьмется из exe файла.).
6. О всех действиях ты узнаешь(если захочешь) сам(посмотрев ключи других программ, например, ключи .ini файла, или .txt, или др. файлов), я скажу тебе о действии "Открыть". В ключе "cdkfile" создаешь подключи:
HKEY_CLASSES_ROOT
- cdkfile
- shell
- open
- command

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

Private Sub Command1_Click()
' Создаем файл с расширением *.CDK, ну это чтобы убедится что новый тип создан.
Open "New_Tip.cdk" For Output As #1
Print #1, "Новый тип"
Close #1

' Создаем основные ключи
CreateNewKey HKEY_CLASSES_ROOT, ".cdk"' Для чего этот ключ нужен, смотри выше(№ 1)
CreateNewKey HKEY_CLASSES_ROOT, "cdkfile\DefaultIcon"' Этот ключ для иконки
CreateNewKey HKEY_CLASSES_ROOT, "cdkfile\shell\open\command"' Этот ключ для действия откорыть

' Записываем значения строковых параметров:
SetKeyValue HKEY_CLASSES_ROOT, "cdkfile", "", "Новый тип *.CDK", REG_SZ ' Смотри № 4:
SetKeyValue HKEY_CLASSES_ROOT, "cdkfile\DefaultIcon", "", App.Path & "\" & "1.exe", REG_SZ ' Записываем путь к иконке, иконка берется из файла 1.exe, который находится в той же папке где и сама программа
SetKeyValue HKEY_CLASSES_ROOT, ".cdk", "", "cdkfile", REG_SZ ' Записываем путь к ключу, смотри № 2
SetKeyValue HKEY_CLASSES_ROOT, "cdkfile\shell\open\command", "", App.Path & "\" & "1.exe", REG_SZ ' Вместо App.Path & "\" & "1.exe" укажи путь к своей программе(№ 6)
End Sub

Теперь перезагрузи комп(без этого иконка не появтся, т.к. она только после перезагрузки появляется), и любуйся.
Исходник.

автор учебника: Падре
Дата создания: 21 Марта 2004 года

Место под Банеры

Логотипы, Банеры, Сайты НА ЗАКАЗ