|
|
(не показано 17 промежуточных версий этого же участника) |
Строка 1: |
Строка 1: |
− | === Как узнать, что пользователь меняет текст в приложении на основе CRichEditView?===
| + | #REDIRECT [[FAQ:WinAPI_VCPP]] |
− | | + | |
− | Нужно добавить обработчик сообщения EN_CHANGE. Сообщение генерируется CRichEditView каждый раз, когда пользователь изменяет текст в окне CRichEditView.
| + | |
− | | + | |
− | ===Чем отличается метод CArray::GetSize от метода CArray::GetCount?===
| + | |
− | | + | |
− | Эти методы абсолютно идентичны. Скорее всего, в более старых версиях библиотек функции, одинаковые по смыслу, имели разные имена. Поэтому в целях совместимости разработчики оставили обе функции.
| + | |
− | | + | |
− | ===Как загрузить текстовую строку из ресурса?===
| + | |
− | | + | |
− | Пример:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | CString m_Temp;
| + | |
− | | + | |
− | m_Temp.LoadString(ID_MYSTRING);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как поменять иконку у элемента item в CListCtrl?===
| + | |
− | | + | |
− | Пример:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // заполняем структуру LVITEM
| + | |
− | LVITEM lvItem;
| + | |
− | | + | |
− | memset(&lvItem, 0, sizeof(lvItem));
| + | |
− |
| + | |
− | lvItem.mask = LVIF_IMAGE; // меняться будет картинка
| + | |
− | lvItem.iItem = ...; // индекс элемента списка (Zero-based)
| + | |
− | lvItem.iSubItem = 0;
| + | |
− | lvItem.iImage = ...; //индекс иконки (из списка элемента)
| + | |
− |
| + | |
− | pList->SetItem(&lvItem);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Имеется класс Class1, содержащий член типа "Class2*". А в Class2 имеется член типа "Class1*". Компилятор выдаёт ошибку. Что надо делать?===
| + | |
− | | + | |
− | Перекрёстное определение обходится следующим образом: код обоих классов разносится в пары файлов *.h и *.cpp, а перед каждым классом прописывается предопределение другого класса:
| + | |
− | | + | |
− | Файл "class1.h":
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | class Class2; // предопределение Class2
| + | |
− | | + | |
− | class Class1
| + | |
− | {
| + | |
− | Class2* m_p2; // компилятор разрешит объявление указателя Class2*
| + | |
− | | + | |
− | void F1(Class2* p); //компилятор разрешит объявление указателя Class2*
| + | |
− | void F2(Class1* p);
| + | |
− | };
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Файл "class1.cpp":
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | #include "class1.h" | + | |
− | #include "class2.h"
| + | |
− | | + | |
− | void Class1::F1(Class2* p)
| + | |
− | {
| + | |
− | // ...
| + | |
− | }
| + | |
− | | + | |
− | void Class1::F2(Class1* p)
| + | |
− | {
| + | |
− | // ...
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Файл "class2.h":
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | class Class1; // предопределение Class1
| + | |
− | | + | |
− | Class2
| + | |
− | {
| + | |
− | Class1* m_p1; // компилятор разрешит объявление указателя Class1*
| + | |
− | | + | |
− | void F3(Class1* p); // компилятор разрешит объявление указателя Class1*
| + | |
− | };
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Файл "class2.cpp":
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | #include "class2.h"
| + | |
− | #include "class1.h"
| + | |
− | | + | |
− | void Class2::F3(Class1* p)
| + | |
− | {
| + | |
− | // ...
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | После предопределения класса компилятор позволяет объявлять указатель на класс, так как переменная указателя на любой тип имеет всегда один и тот же размер (4 байта).
| + | |
− | | + | |
− | ===Как переключить раскладку клавиатуры в другом (активном) процессе?===
| + | |
− | | + | |
− | Для этого можно использовать функции:
| + | |
− | * GetKeyboardLayout - определить текущую раскладку
| + | |
− | * LoadKeyboardLayout - загрузить новую раскладку
| + | |
− | * VerLanguageName - получить строку с описанием языка
| + | |
− | | + | |
− | Подробности - в MSDN.
| + | |
− | | + | |
− | ===Как получить хендл элемента управления, зная его идентификатор?===
| + | |
− | | + | |
− | Пример: пусть элемент Edit лежит на окне CMyWnd.
| + | |
− | Тогда:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | void CMyWnd::некая_процедура()
| + | |
− | {
| + | |
− | // для MFC
| + | |
− | CEdit* ed = (CEdit*)GetDlgItem(ID_EDIT1);
| + | |
− | HWND hwnd = ed->GetSafeHwnd();
| + | |
− | if (hwnd)
| + | |
− | {
| + | |
− | // ...
| + | |
− | }
| + | |
− |
| + | |
− | // для Win32 API
| + | |
− | HWND hwnd = GetDlgItem(m_hWnd, ID_EDIT1);
| + | |
− | if (hwnd)
| + | |
− | {
| + | |
− | // ...
| + | |
− | }
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как из дочернего окна закрыть приложение?===
| + | |
− | | + | |
− | Для этого нужно послать родительскому окну сообщение WM_CLOSE.
| + | |
− | | + | |
− | Через указатель на родительское окно:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | pParent->PostMessage(WM_CLOSE);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | При помощи хендла родительского окна:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | ::PostMessage(pParent->m_hWnd, WM_CLOSE, (WPARAM)0, (LPARAM)0);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Примечание: WM_CLOSE аналогично нажатию на крестик окна. То есть, это "мягкое" закрытие окна, так как это сообщение приложение может обработать по своему.
| + | |
− | Если же точно так же отправить сообщение WM_QUIT (при этом в wParam указывается значение, возвращаемое процессом после жавершения), то оконная процедура, получив это сообщение, просто прекращает работу и приложение закрывается сразу.
| + | |
− | | + | |
− | ===Как корректно перевести тип BSTR в CString и наоборот?===
| + | |
− | | + | |
− | # Для конвертирования BSTR в CString нужно просто присвоить переменной типа CString переменную типа BSTR. Оператор "=" класса CString сам выполнит всю работу.
| + | |
− | # Для конвертирования BSTR в CString нужно выполнить код:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // указатель на будущий буфер с BSTR.
| + | |
− | // По сути, BSTR определён как typedef WCHAR* BSTR;
| + | |
− | // то есть - указатель на 16-битный символ юникод.
| + | |
− | BSTR bstrHE=0;
| + | |
− | | + | |
− | // строка для конвертации
| + | |
− | CString Str="мой текст";
| + | |
− | | + | |
− | // Выделяем память и загружаем адрес блока в bstrHE
| + | |
− | bstrHE=Str.AllocSysString();
| + | |
− | | + | |
− | // ...
| + | |
− | // тут работаем с bstrHE[]
| + | |
− | // ...
| + | |
− | | + | |
− | // освобождаем память
| + | |
− | SysFreeString(bstrHE);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как получить доступ к графическим ресурсам элементов текущей темы оформления Windows?===
| + | |
− | | + | |
− | Для этого нужно использовать функции:
| + | |
− | * OpenThemeData
| + | |
− | * DrawThemeBackground
| + | |
− | * DrawFrameControl
| + | |
− | * CloseThemeData
| + | |
− | | + | |
− | Подробности - в MSDN.
| + | |
− | | + | |
− | ===Имеется класс, производный от CDialog. Когда диалог в фокусе, нажатие на Enter или Esc приводит к закрытию диалога. Как это запретить?===
| + | |
− | | + | |
− | При нажатии Enter происходит выполнение команды IDOK, при нажатии Esc - IDCANCEL.
| + | |
− | Нужно переопределить в классе виртуальные функции OnOk() и OnCancel(), которые соответственно вызываются для этих команд. При помощи визарда эти функции добавить можно следующим образом.
| + | |
− | | + | |
− | Поместить на диалог две кнопки с идентификаторами IDOK и IDCANCEL (при создании нового диалога они там есть сразу), двойной щелчок по кнопке добавит соответствующий обработчик. В обработчиках надо удалить вызовы СDialog::OnOK() и CDialog::OnCancel(), тогда диалог закрываться не будет.
| + | |
− | | + | |
− | Однако тогда диалог станет невозможно закрыть кнопкой с крестиком. Это обходится так: добавляем обработчик сообщения WM_CLOSE - OnClose(), и в нём делаем вызов OnOk() или OnCancel():
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | void CPlayersPropsDialog::OnClose()
| + | |
− | {
| + | |
− | CDialog::OnClose();
| + | |
− | CDialog::OnCancel(); // добавлено
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Я написал в VC++ простейший макрос, где невозможно ошибиться, а компилятор всё же выдаёт ошибку.===
| + | |
− | | + | |
− | Скорее всего после одного из символов соединения строк "\" имеется пробел или табуляция. Их надо удалить. Можно найти их "на ощупь", а можно включить показ непечатных символов в студии. Найти эту команду можно так:
| + | |
− | | + | |
− | Tools->Customize...->вкладка Commands. В окошке Category выбрать "Edit", и среди кнопок справа найти кнопку, на которой написано "a.b". Эту кнопку надо перенести на одну из панелей инструментов студии.
| + | |
− | | + | |
− | ===Как получить список всех процессов, включая idle?===
| + | |
− | | + | |
− | Использовать функции:
| + | |
− | * Process32First
| + | |
− | * Precess32Next
| + | |
− | | + | |
− | Подробности - в MSDN.
| + | |
− | | + | |
− | ===Как программно установить переменные окружения?===
| + | |
− | | + | |
− | Чтобы добавить или изменить их программно, необходимо воспользоваться ключом реестра
| + | |
− | | + | |
− | HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\Session Manager\Environment
| + | |
− | | + | |
− | Затем нужно отправить широковещательное сообщение WM_SETTINGCHANGE - это позволит приложениям узнать об изменениях:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | ::SendMessage(HWND_BROADCAST, WM_SETTINGCHANGE, 0, 0);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как сделать, чтобы у окна был черный фон?===
| + | |
− | | + | |
− | Нужно переопределить обработчик сообщения WM_CTLCOLOR - OnCtlColor():
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // глобальная переменная или член класса CMyDlg,
| + | |
− | // инициализированный в OnInitDialog()
| + | |
− | CBrush br(RGB(0, 0, 0));
| + | |
− | | + | |
− | HBRUSH CMyDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
| + | |
− | {
| + | |
− | HBRUSH hbr = CMyDlg::OnCtlColor(pDC, pWnd, nCtlColor);
| + | |
− |
| + | |
− | // фон диалога (или View)
| + | |
− | if(nCtlColor == CTLCOLOR_DLG)
| + | |
− | {
| + | |
− | return (HBRUSH)br;
| + | |
− | }
| + | |
− |
| + | |
− | // фон элементов SCtatic, лежащих на форме (если нужно)
| + | |
− | if(nCtlColor == CTLCOLOR_STATIC)
| + | |
− | {
| + | |
− | // делаем фон текста статика прозрачным
| + | |
− | pDC->SetBkMode(TRANSPARENT);
| + | |
− | return (HBRUSH)br;
| + | |
− | }
| + | |
− |
| + | |
− | return hbr;
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Почему на моем компьютере экзешник, созданный в MFC запускается, а на других компьютерах - нет? Требует какую-то dll.===
| + | |
− | | + | |
− | Нужно выбирать режим Release: Menu->Build->Configurations...->Release.
| + | |
− | Экзешник, соответственно, брать из папки Release проекта.
| + | |
− | | + | |
− | ===Как в диалог добавить меню?===
| + | |
− | | + | |
− | Нужно создать в редакторе ресурсов новое меню, затем вставить его в диалог:
| + | |
− | # Открыть в редакторе диалог;
| + | |
− | # В свойствах (во вкладке General, окошко Menu) указать идентификатор ресурса меню;
| + | |
− | # При помощи визарда добавить для меню обработчики OnCommand() и OnUpdateСommand().
| + | |
− | | + | |
− | ===Почему не вызывается OnUpdate для пунктов меню (не получается ни затенить, ни отметку поставить)?===
| + | |
− | | + | |
− | В диалоге (CDialog) у меню апдейт не происходит сам, в отличие от мейнфрейма (CFrameWnd). Для этого нужно сделать апдейт самим, в обработчике сообщения WM_KICKIDLE.
| + | |
− | | + | |
− | Визардом этот обработчик не добавляется, поэтому можно переопределить виртуальную DefWindowProc() и там обработать:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | #include <afxpriv.h> // этот файл надо включить, там определёна WM_KICKIDLE
| + | |
− | | + | |
− | LRESULT CMyDialog::DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam)
| + | |
− | {
| + | |
− | if(message == WM_KICKIDLE)
| + | |
− | {
| + | |
− | // делаем апдейт для всех пунктов меню
| + | |
− | CMenu* pMainMenu = GetMenu();
| + | |
− | if(pMainMenu)
| + | |
− | {
| + | |
− | CCmdUI cmdUI;
| + | |
− | for(UINT n=0; n<pMainMenu->GetMenuItemCount(); n++)
| + | |
− | {
| + | |
− | CMenu* pSubMenu=pMainMenu->GetSubMenu(n);
| + | |
− | if(!pSubMenu)
| + | |
− | continue;
| + | |
− |
| + | |
− | cmdUI.m_nIndexMax = pSubMenu->GetMenuItemCount();
| + | |
− | for(UINT i=0; i<cmdUI.m_nIndexMax; i++)
| + | |
− | {
| + | |
− | cmdUI.m_nIndex = i;
| + | |
− | cmdUI.m_nID = pSubMenu->GetMenuItemID(i);
| + | |
− | cmdUI.m_pMenu = pSubMenu;
| + | |
− | cmdUI.DoUpdate(this, FALSE);
| + | |
− | }
| + | |
− | }
| + | |
− | }
| + | |
− | }
| + | |
− | | + | |
− | return CDialog::DefWindowProc(message, wParam, lParam);
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как в главном окне отключить системное меню (в левом верхнем углу) и кнопки (в правом верхнем углу)?===
| + | |
− | | + | |
− | При создании окна нужно убрать стиль WS_SYSMENU. Теперь пользователь корректно может закрыть окно только Alt+F4, ну, или если вы предоставите ему дополнительную возможность сделать это.
| + | |
− | | + | |
− | Пример 1. (для SDI, MDI)
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // этот обработчик уже добавлен визардом
| + | |
− | BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)
| + | |
− | {
| + | |
− | if(!CFrameWnd::PreCreateWindow(cs))
| + | |
− | return FALSE;
| + | |
− | // TODO: Modify the Window class or styles here by modifying
| + | |
− | // the CREATESTRUCT cs
| + | |
− | | + | |
− | ///////////////////////////////
| + | |
− | // добавленная часть
| + | |
− | {
| + | |
− | // убираем стиль
| + | |
− | cs.style& =~ WS_SYSMENU;
| + | |
− | }
| + | |
− | ///////////////////////////////
| + | |
− | | + | |
− | return TRUE;
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Пример 2. (для диалога)
| + | |
− | Если диалог описан в ресурсах, то в свойствах диалога убрать галочки:
| + | |
− | * "System menu"
| + | |
− | * "Minimize box"
| + | |
− | * "Maximize box"
| + | |
− | | + | |
− | ===Как запретить пользователю нажать на кнопку?===
| + | |
− | | + | |
− | Нужно применить метод класса CWnd::EnableWindow(...) для кнопки. Значение 0 параметра делает кнопку неактивной, 1 - делает активной (кстати, не только для кнопки можно, а для любого класса, производного от класса CWnd).
| + | |
− | | + | |
− | Пусть имеется некий диалог, на нём лежит кнопка c ID == IDC_1. В любом месте кода диалога (кроме конструктора и деструктора, хотя, если проверить наличие валидного хендла диалога, как в примере, то ничего страшного не будет и там) выполняем код:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // делаем кнопку неактивной
| + | |
− | if(m_hWnd)
| + | |
− | {
| + | |
− | CWnd* pw = 0;
| + | |
− | pw = GetDlgItem(IDC_1);
| + | |
− | if(pw)
| + | |
− | {
| + | |
− | pw->EnableWindow(0);
| + | |
− | }
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Что означают сообщения, которые студия выводит при компиляции, вроде таких: Loaded 'C:\WINDOWS\SYSTEM\WSOCK32.DLL', no matching symbolic information found?===
| + | |
− | | + | |
− | Это строка говорит о том, что студия не смогла найти отладочные символы для WSOCK32.DLL. В этом нет ничего страшного. Просто, при наличии отладочных символов в процессе отладки, можно получить доступ к именам функции из системных DLL. Например, вместо ''KERNEL32! 0x77E8B184()'' увидим ''KERNEL32!CreateThread''.
| + | |
− | | + | |
− | ===Как обработать сообщения, которые приходят к элементу управления?===
| + | |
− | | + | |
− | Скажем, нужно обработать сообщения, приходящие к элементу класса CEdit. Нужно произвести от CEdit свой класс и в виртуальной процедуре класса WindowProc перехватить нужные сообщения.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | virtual LRESULT WindowProc(
| + | |
− | UINT message,
| + | |
− | WPARAM wParam,
| + | |
− | LPARAM lParam
| + | |
− | );
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Пример. Полностью выключаем, скажем, обработку сообщения WM_CHAR.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | LRESULT CMyEdit::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
| + | |
− | {
| + | |
− | switch(message)
| + | |
− | {
| + | |
− | case WM_CHAR:
| + | |
− | {
| + | |
− | return 0;
| + | |
− | }
| + | |
− | break;
| + | |
− | }
| + | |
− |
| + | |
− | return CEdit::WindowProc(message, wParam, lParam);
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Чтобы связать экземпляр класса с элементом управления на форме, добавьте визардом для
| + | |
− | элемента управления переменную класса (выбрав не Value, а Control).
| + | |
− | | + | |
− | ===Как настроить количество пробелов в табуляции?===
| + | |
− | | + | |
− | Есть пара способов:
| + | |
− | # Tools->Options->Tabs->Insert Spaces
| + | |
− | # Правой кнопкой мыши щёлкнуть по тексту, в контекстном меню выбрать properties.
| + | |
− | | + | |
− | ===Как автоматически расставить отступы?===
| + | |
− | | + | |
− | Нужно в студии выделить текст для форматирования и нажать Alt+F8.
| + | |
− | | + | |
− | Ещё есть программа Artistic Style ( http://sourceforge.net/projects/astyle/ ), которая занимается переформатированием кода.
| + | |
− | | + | |
− | ===Поиск границ блока.===
| + | |
− | | + | |
− | Сочетание клавиш CTRL+"}" ищет парную фигурную скобку в тексте и переходит к ней.
| + | |
− | | + | |
− | ===Вертикальное выделение текста.===
| + | |
− | | + | |
− | Если удерживать клавишу Alt, а затем начать выделять текст мышкой, то область выделения принимает форму прямоугольника, что позволяет выделять вертикальные фрагменты текста.
| + | |
− | | + | |
− | ===Как узнать количество установленных в CListCtrl столбцов?===
| + | |
− | | + | |
− | Пример:
| + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // указатель на переменную-элемент управления
| + | |
− | CListCtrl* pL = ...;
| + | |
− | | + | |
− | // получаем количество столбцов
| + | |
− | int count = pL->GetHeaderCtrl()->GetItemCount();
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как получить иконку приложения?===
| + | |
− | | + | |
− | Если приложение запущено, то нужно найти его главное окно и послать ему сообщение WM_GETICON.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // функция возвращант хендл иконки
| + | |
− | LRESULT SendMessage(hWnd, WM_GETICON, wParam, 0);
| + | |
− | | + | |
− | // hWnd == хендл окна приложения
| + | |
− | // wParam ==
| + | |
− | // 1)==ICON_BIG - получить большую иконку
| + | |
− | // 2)==ICON_SMALL - получить маленькую иконку
| + | |
− | // 3)==ICON_SMALL2 - получить маленькую иконку, если она определена
| + | |
− | // в приложении. Если её нет, то маленькую иконку,
| + | |
− | // сгенерированную системой из большой иконки
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как в CString можно найти или вырезать часть строки?===
| + | |
− | | + | |
− | Это можно сделать при помощи методов класса CString:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | // вырезает кусок строки
| + | |
− | CString Mid(int nFirst) const;
| + | |
− | CString Mid(int nFirst, int nCount) const;
| + | |
− | | + | |
− | // возвращает кусок строки (сначала или с конца)
| + | |
− | CString Left(int nCount) const;
| + | |
− | CString Right(int nCount) const;
| + | |
− | | + | |
− | // возвращает начальный кусок строки, в котором есть только
| + | |
− | // символы из набора, представленного в lpszCharSet
| + | |
− | CString SpanIncluding(LPCTSTR lpszCharSet) const;
| + | |
− | | + | |
− | // возвращает начальный кусок строки, в котором нет
| + | |
− | // символов из набора, представленного в lpszCharSet
| + | |
− | CString SpanExcluding(LPCTSTR lpszCharSet) const;
| + | |
− | | + | |
− | // убирает "пробелоподобные" символы из самого начала строки.
| + | |
− | // то есть - пробел, табуляцию (\t), возврат каретки, перевод строки (/r/n)
| + | |
− | void TrimLeft();
| + | |
− | | + | |
− | // убирает все повторы символа из самого начала строки
| + | |
− | void TrimLeft(TCHAR chTarget);
| + | |
− | | + | |
− | // убирает из самого начала строки все символа из набора lpszTargets
| + | |
− | void TrimLeft(LPCTSTR lpszTargets);
| + | |
− | void TrimRight();
| + | |
− | void TrimRight(TCHAR chTarget);
| + | |
− | void TrimRight(LPCTSTR lpszTargets);
| + | |
− | | + | |
− | // поиск в строке
| + | |
− | int Find(TCHAR ch) const;
| + | |
− | int Find(LPCTSTR lpszSub) const;
| + | |
− | int Find(TCHAR ch, int nStart) const;
| + | |
− | int Find(LPCTSTR pstr, int nStart) const;
| + | |
− | | + | |
− | // поиск, начиная с конца
| + | |
− | int ReverseFind(TCHAR ch) const;
| + | |
− | | + | |
− | // поиск позиции первого символа, одного из набора,
| + | |
− | // представленного в lpszCharSet
| + | |
− | int FindOneOf(LPCTSTR lpszCharSet) const;
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как отобразить на элементах управления промежуточные результаты длительных вычислений?===
| + | |
− | | + | |
− | В общем случае это делается принудительной перерисовкой нужного окна путём объявления этого окна невалидным.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | pWnd->Invalidate(0);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Затем непосредственной отсылки в оконную процедуру сообщения WM_PAINT.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | pWnd->UpdateWindow();
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Пример:
| + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | CWnd* pWnd = ...; // окно, которое надо перерисовывать
| + | |
− | | + | |
− | for(int i=0; i<10000; i++)
| + | |
− | {
| + | |
− | // меняется содержимое окна
| + | |
− | // ...
| + | |
− | | + | |
− | // немедленная перерисовка
| + | |
− | pWnd->Invalidate(0);
| + | |
− | pWnd->UpdateWindow();
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Также можно также просто вызвать метод RedrawWindow() с параметрами по умолчанию.
| + | |
− | | + | |
− | ===Я написал DLL, которую используют несколько приложений. Всё работает, но когда происходит очередной вызов функции из DLL, почему то, данные в функции обнуляются. С чем это связано?===
| + | |
− | | + | |
− | Переменные и массивы в DLL, содержимое которых должно использоваться несколькими процессами, должны объявляться статическими. Статическая переменная или массив инициализируются только один раз в момент загрузки DLL.
| + | |
− | | + | |
− | Пример:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | DWORD calltest()
| + | |
− | {
| + | |
− | // будет выполнено только при первом вызове
| + | |
− | static DWORD callcount = 0;
| + | |
− | static DWORD str[1000] = {0};
| + | |
− | | + | |
− | // будет выполняться каждый раз
| + | |
− | callcount++;
| + | |
− | return callcount;
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как открыть проекцию файла в память и как с ней работать?===
| + | |
− | | + | |
− | Создание проекции:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | HANDLE CreateFileMapping(
| + | |
− | HANDLE hFile, // хендл уже открытого файла
| + | |
− | LPSECURITY_ATTRIBUTES lpAttributes,
| + | |
− | DWORD flProtect, // способ открытия проекции
| + | |
− | DWORD dwMaximumSizeHigh, // размер файла (старшие 4 байта, обычно - 0)
| + | |
− | DWORD dwMaximumSizeLow, // размер файла (младшие 4 байта)
| + | |
− | LPCTSTR lpName
| + | |
− | );
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Доступ к созданной проекции производится процедурой:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | LPVOID MapViewOfFile(
| + | |
− | HANDLE hFileMappingObject, // хендл проекции
| + | |
− | DWORD dwDesiredAccess, // способ работы с проекцией
| + | |
− | DWORD dwFileOffsetHigh,
| + | |
− | DWORD dwFileOffsetLow, // смещение от начала файла (младшие 4 байта)
| + | |
− | SIZE_T dwNumberOfBytesToMap // длина в байтах (если 0 - то весь файл)
| + | |
− | );
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Далее с файлом можно работать как с обычным массивом в памяти.
| + | |
− | | + | |
− | После работы с проекцией, её надо освободить.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | BOOL UnmapViewOfFile(
| + | |
− | LPCVOID lpBaseAddress // адрес, который вернула процедура MapViewOfFile
| + | |
− | );
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | Пример:
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | HANDLE hFile = ...; // хендл уже открытого файла
| + | |
− | DWORD dwdFileLen = ...; // размер открытого файла
| + | |
− | HANDLE hMapFile = 0; // хендл для проекции
| + | |
− | | + | |
− | // создаём проекцию "только для чтения"
| + | |
− | hMapFile = ::CreateFileMapping(hFile, 0, PAGE_READONLY, 0, dwdFileLen, 0);
| + | |
− | | + | |
− | // хендл hFile можно закрыть уже здесь, в принципе,
| + | |
− | // но мы сделаем это позже
| + | |
− | | + | |
− | // получаем доступ к проекции (тоже только для чтения)
| + | |
− | BYTE* pbyFile = (BYTE*)::MapViewOfFile(hMapFile, FILE_MAP_READ, 0, 0, 0);
| + | |
− | | + | |
− | ///////////////////
| + | |
− | // тут работаем с массивом pbyFile[dwdFileLen] как с обычным,
| + | |
− | // только не забываем, что он открыт только для чтения
| + | |
− | // ...
| + | |
− | // ...
| + | |
− | // ...
| + | |
− | ///////////////////
| + | |
− | | + | |
− | // отключаем файл данных от адресного пространства
| + | |
− | UnmapViewOfFile(pbyFile);
| + | |
− | | + | |
− | // освобождаем хендл проекции
| + | |
− | CloseHandle(hMapFile)
| + | |
− | | + | |
− | // освобождаем хендл открытого файла
| + | |
− | CloseHandle(hFile);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как в MFC Grid control отобразить картинку в ячейке?===
| + | |
− | | + | |
− | Нужно создать список изображений (CImageList), затем передать указатель на список в таблицу. Поскольку элемент управления лишь копирует указатель, список должен создаваться динамически, либо быть членом класса, экземпляр которого существует всё время работы таблицы.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | CImageList m_ImageList;
| + | |
− | CGridCtrL m_Grid1;
| + | |
− | | + | |
− | // ...
| + | |
− | | + | |
− | // неким образом создаётся список
| + | |
− | m_ImageList.Create(...);
| + | |
− | | + | |
− | // ...
| + | |
− | | + | |
− | //вставляем в элемент управления
| + | |
− | m_Grid1.SetImageList(&m_ImageList);
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как при помощи IPicture отобразить картинку из файла? ===
| + | |
− | | + | |
− | Пример - несложная функция, отображающая картинку из файла. Поддерживаются форматы BMP, GIF, JPEG, PNG, TIFF, EMF.
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | #include "atlconv.h"
| + | |
− | #define HandleIsValid(H) (H != (HANDLE) - 1 && H != (HANDLE)0)
| + | |
− | | + | |
− | // *pDestDC - CDC, на который предполагается вывод картинки
| + | |
− | // pchzFilePath - путь к файлу
| + | |
− | // pWid - (возвращает) ширина картинки
| + | |
− | // pHig - (возвращает) высота картинки
| + | |
− | bool DrawBitmapFromFile(CDC* pDestDC, const char* pchzFilePath, int *pWid = 0, int *pHig = 0)
| + | |
− | {
| + | |
− | bool result = false;
| + | |
− | | + | |
− | if (pWid)
| + | |
− | *pWid = 0;
| + | |
− | | + | |
− | if (pHig)
| + | |
− | *pHig = 0;
| + | |
− | | + | |
− | int Wid = 0;
| + | |
− | int Hig = 0;
| + | |
− | | + | |
− | IPicture* pPic = 0;
| + | |
− | try
| + | |
− | {
| + | |
− | IPicture* ptmpPic = 0;
| + | |
− | USES_CONVERSION;
| + | |
− | HRESULT hr;
| + | |
− | | + | |
− | CString txt = pchzFilePath;
| + | |
− | hr = ::OleLoadPicturePath(
| + | |
− | const_cast<LPOLESTR>(T2COLE(txt)),
| + | |
− | 0,
| + | |
− | 0,
| + | |
− | 0,
| + | |
− | IID_IPicture,
| + | |
− | reinterpret_cast<void **>(&ptmpPic)
| + | |
− | );
| + | |
− | | + | |
− | if(hr == S_OK && ptmpPic)
| + | |
− | {
| + | |
− | pPic = ptmpPic;
| + | |
− | }
| + | |
− | else
| + | |
− | {
| + | |
− | throw 0;
| + | |
− | }
| + | |
− | | + | |
− | OLE_XPOS_HIMETRIC cxSrc;
| + | |
− | OLE_YPOS_HIMETRIC cySrc;
| + | |
− | | + | |
− | if(S_OK != pPic->get_Width(&cxSrc))
| + | |
− | {
| + | |
− | throw 0;
| + | |
− | }
| + | |
− | | + | |
− | if(S_OK != pPic->get_Height(&cySrc))
| + | |
− | {
| + | |
− | throw 0;
| + | |
− | }
| + | |
− | | + | |
− | Wid = cxSrc / 26;
| + | |
− | Hig = cySrc / 26;
| + | |
− | | + | |
− | // рисуем
| + | |
− | if(S_OK != pPic->Render(pDestDC->GetSafeHdc(),
| + | |
− | 0, Hig, Wid, -Hig, 0, 0, cxSrc, cySrc, 0))
| + | |
− | {
| + | |
− | throw 0;
| + | |
− | }
| + | |
− | }
| + | |
− | catch (...)
| + | |
− | {
| + | |
− | result = false;
| + | |
− | }
| + | |
− | | + | |
− | if(pPic)
| + | |
− | {
| + | |
− | pPic->Release();
| + | |
− | pPic = 0;
| + | |
− | }
| + | |
− | | + | |
− | if(pWid)
| + | |
− | {
| + | |
− | *pWid = Wid;
| + | |
− | }
| + | |
− | | + | |
− | if(pHig)
| + | |
− | {
| + | |
− | *pHig = Hig;
| + | |
− | }
| + | |
− | | + | |
− | return result;
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как позволить пользователю начать ввод новой строки в многострочном поле редактирования?===
| + | |
− | | + | |
− | Нужно в редакторе ресурсов поставить для свойств окошка CEdit галочку "WantReturn" или добавить стиль ES_WANTRETURN при создании CEdit элемента управления динамически. Тогда при нажатии Enter в поле редактирования будет вводиться новая строка.
| + | |
− | | + | |
− | ===Как выводят картинку-логотип (splash screen) при запуске программы?===
| + | |
− | | + | |
− | Для этого используется класс, производный от CDialog. Этот диалог после создания показывает себя на экран, выводя на себя рисунок. Также запускает таймер, по срабатывани которого диалог гасится. Создаётся диалог в процедуре InitInstance приложения.
| + | |
− | | + | |
− | В VC6++ также есть уже готовый компонент: Project->Add To Project -> Components and Controls->(в папке "Visual C++ Components" есть SplashScreen.).
| + | |
− | | + | |
− | ===Как сделать, чтобы при выпадении списка у ComboBox была не одна строка, а больше? Вроде все свойства покрутил, не помогает: вместо выпадающего списка - одна строка и скролл.===
| + | |
− | | + | |
− | В редакторе ресурсов надо щелкнуть в комбобоксе на треугольнике - появляется рамка для растягивания вниз - размер выпадающего списка.
| + | |
− | | + | |
− | ===VC++.NET. На форму установил ActiveX ListView. Как объявить класс и переменную для этого элемента?===
| + | |
− | | + | |
− | Щёлкнуть правой кнопкой мыши на элементе, в меню - add variable или add class.
| + | |
− | | + | |
− | ===Как работать с буфером обмена (Clipboard)?===
| + | |
− | | + | |
− | ====КОПИРОВАНИЕ В БУФЕР ОБМЕНА====
| + | |
− | | + | |
− | Алгоритм:
| + | |
− | # Готовим данные
| + | |
− | #* Выделяем память из кучи, вызывая GlobalAlloc()
| + | |
− | #* Получаем указатель на выделенную память, вызывая GlobalLock()
| + | |
− | #* Заполняем данные
| + | |
− | #* Освобождаем указатель, вызывая GlobalUnlock();
| + | |
− | # Открываем буфер обмена, вызывая OpenClipboard();
| + | |
− | # Очищаем буфер, вызывая EmptyClipboard();
| + | |
− | # Вызываем SetClipboardData() один раз для каждого формата вставляемых данных (имеется в виду - если одни и те же данные представлены в разных форматах, и приложение может эти форматы создать)
| + | |
− | # Закрываем буфер, вызывая CloseClipboard();
| + | |
− | # ВЫЗЫВАТЬ GlobalFree() НЕ НУЖНО, в этом случае это предоставлено системе
| + | |
− | | + | |
− | Скажем, хотим поместить в буфер обмена текст.
| + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | //готовим данные
| + | |
− | //выделяем из кучи память для данных
| + | |
− | HANDLE hglbCopy = GlobalAlloc(GMEM_MOVEABLE,10);
| + | |
− | if(!hglbCopy)
| + | |
− | return;
| + | |
− | | + | |
− | //получаем указатель
| + | |
− | void* lpStr = GlobalLock(hglbCopy);
| + | |
− | if(!lpStr)
| + | |
− | return;
| + | |
− | | + | |
− | //заполняем данные
| + | |
− | strcpy((char*)lpStr,"my text");
| + | |
− | | + | |
− | //освобождаем указатель
| + | |
− | GlobalUnlock(hglbCopy);
| + | |
− | lpStr=0;
| + | |
− | | + | |
− | //открываем буфер обмена
| + | |
− | OpenClipboard();
| + | |
− | //очищаем
| + | |
− | EmptyClipboard();
| + | |
− | | + | |
− | //текстовый формат
| + | |
− | SetClipboardData(CF_TEXT,hglbCopy);
| + | |
− | | + | |
− | //закрываем буфер
| + | |
− | CloseClipboard();
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | | + | |
− | | + | |
− | ====ИЗВЛЕЧЕНИЕ ИЗ БУФЕРА ОБМЕНА====
| + | |
− | | + | |
− | Алгоритм:
| + | |
− | # Проверяем, что поддерживается нужный формат данных, вызывая IsClipboardFormatAvailable()
| + | |
− | # Открываем буфер обмена, вызывая OpenClipboard();
| + | |
− | # Достаём данные
| + | |
− | #* Получаем из буфера хендл требуемого формата, вызывая GetClipboardData()
| + | |
− | #* Получаем указатель на выделенную память, вызывая GlobalLock()
| + | |
− | #* Работаем с данными
| + | |
− | #* Освобождаем указатель, вызывая GlobalUnlock();
| + | |
− | # Закрываем буфер, вызывая CloseClipboard();
| + | |
− | | + | |
− | Скажем, хотим извлечь из буфера обмена текст.
| + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | //Проверяем, что поддерживается нужный формат данных,
| + | |
− | if(!IsClipboardFormatAvailable(CF_TEXT))
| + | |
− | return;
| + | |
− | | + | |
− | //открываем буфер обмена
| + | |
− | OpenClipboard();
| + | |
− | | + | |
− | //достаём данные
| + | |
− | | + | |
− | //текстовый формат
| + | |
− | HANDLE hglbCopy = GetClipboardData(CF_TEXT);
| + | |
− | if(!hglbCopy)
| + | |
− | return;
| + | |
− | | + | |
− | //получаем указатель
| + | |
− | void* lpStr = GlobalLock(hglbCopy);
| + | |
− | if(!lpStr)
| + | |
− | return;
| + | |
− | | + | |
− | //читаем данные
| + | |
− | char data[10];
| + | |
− | memmove(data,lpStr,sizeof(data));
| + | |
− | | + | |
− | //освобождаем указатель
| + | |
− | GlobalUnlock(hglbCopy);
| + | |
− | lpStr=0;
| + | |
− | | + | |
− | //закрываем буфер
| + | |
− | CloseClipboard();
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | | + | |
− | ===Как получить полный путь к экзешнику из самой программы?===
| + | |
− | Можно прочитать параметры командной строки - в командную строку первым параметром система всегда передаёт заключённый в кавычки полный путь к файлу запущенной программы. Достаём путь следующим образом:
| + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | BOOL CMyApp::InitInstance()
| + | |
− | {
| + | |
− | //добыча полного имени экзешника
| + | |
− | CString csFullExeName;
| + | |
− | {
| + | |
− | CString csAppName=GetCommandLine();
| + | |
− | csAppName.Delete(0,1);
| + | |
− | csAppName.Replace('\"','\0');
| + | |
− | csFullExeName=(const char*)csAppName;
| + | |
− | }
| + | |
− | //теперь csFullExeName содержит искомый путь
| + | |
− | | + | |
− | //...
| + | |
− | //...
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Каким способом exe файл может заменить самого себя? ===
| + | |
− | | + | |
− | К примеру, программа должна обновляется через Internet и хочет обновить свой exe-файл. Но так как он в данный момент запущен, это будет, естественно, невозможно сделать напрямую.
| + | |
− | Один из способов - сделать копию экзешника и из него обновиться
| + | |
− | | + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | #define def_exeNAMEup "c:\\myprog_toupdate.exe" //путь к старому файлу
| + | |
− | #define def_exeNAMEdnld "c:\\myprog_downloaded.exe" //путь к новому файлу (уже "скачали":) )
| + | |
− | #define def_keyUdate "/update"
| + | |
− | | + | |
− | void CTESTFAQApp::UpdateItself()
| + | |
− | {
| + | |
− | //добыча полного имени экзешника
| + | |
− | CString csFullExeName;
| + | |
− | {
| + | |
− | CString csAppName=GetCommandLine();
| + | |
− | csAppName.Delete(0,1);
| + | |
− | csAppName.Replace('\"','\0');
| + | |
− | csFullExeName=(const char*)csAppName;
| + | |
− | }
| + | |
− | //теперь csFullExeName - содержит искомый путь
| + | |
− | | + | |
− | //делаем копию экзешника с другим именем
| + | |
− | CopyFile(csFullExeName,def_exeNAMEup,0);
| + | |
− | | + | |
− | //передаём в параметры нового процесса ключ и путь к текущему файлу
| + | |
− | CString csParams=def_keyUdate;
| + | |
− | csParams+=" ";
| + | |
− | csParams+=csFullExeName;
| + | |
− | | + | |
− | //запускаем копию
| + | |
− | STARTUPINFO si;
| + | |
− | PROCESS_INFORMATION pi;
| + | |
− | ZeroMemory(&si,sizeof(si));
| + | |
− | si.cb = sizeof(si);
| + | |
− | ZeroMemory(&pi,sizeof(pi));
| + | |
− | if(CreateProcess(def_exeNAMEup,def_keyUdate,0,0,0,0,0,0,&si,&pi))
| + | |
− | {
| + | |
− | //завершаем текущий процесс
| + | |
− | ExitFromMyApp();
| + | |
− | }
| + | |
− | }
| + | |
− | | + | |
− | void CTESTFAQApp::ExitFromMyApp()
| + | |
− | {
| + | |
− | exit(0);
| + | |
− | }
| + | |
− | | + | |
− | BOOL CTESTFAQApp::InitInstance()
| + | |
− | {
| + | |
− | //смотрим, не запущено ли для апдейта?
| + | |
− | CString txt;
| + | |
− | txt=m_lpCmdLine;
| + | |
− | if(txt.Find(def_keyUdate)==0)
| + | |
− | {
| + | |
− | //делаем айдейт
| + | |
− | txt.Replace(def_keyUdate,"");
| + | |
− | txt.TrimLeft();
| + | |
− | | + | |
− | CString csFullExeName=txt;
| + | |
− | | + | |
− | //берём откуда то уже скачанный файл обновления
| + | |
− | //...
| + | |
− | CString csFull_Downloaded_ExeName=def_exeNAMEdnld;
| + | |
− | | + | |
− | //копируем файл (обновляем старый то есть)
| + | |
− | CopyFile(csFull_Downloaded_ExeName,csFullExeName,0);
| + | |
− | | + | |
− | //запускаем обновлённого мученника
| + | |
− | STARTUPINFO si;
| + | |
− | PROCESS_INFORMATION pi;
| + | |
− | ZeroMemory(&si,sizeof(si));
| + | |
− | si.cb = sizeof(si);
| + | |
− | ZeroMemory(&pi,sizeof(pi));
| + | |
− | if(CreateProcess(csFullExeName,0,0,0,0,0,0,0,&si,&pi))
| + | |
− | | + | |
− | ExitFromMyApp();
| + | |
− | }
| + | |
− | //...
| + | |
− | //...
| + | |
− | }
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | === Как производится конвертация из кодировки UTF8 в 1251 и наоборот?===
| + | |
− | | + | |
− | Можно перевести строку из UTF8 в Unicode, затем из Unicode в 1251
| + | |
− | | + | |
− | Ниже приведена структура, содержащая процедуры перекодирования,
| + | |
− | а также процедуру с примером использования
| + | |
− | <syntaxhighlight lang="cpp">
| + | |
− | //описание структуры
| + | |
− | struct coder
| + | |
− | {
| + | |
− | //utf8->unicode
| + | |
− | static wchar_t* utf8_to_unicode__dontForgetDeleteArr(const char *utf8_string)
| + | |
− | {
| + | |
− | wchar_t* pRes=0;
| + | |
− | int res_len=0;
| + | |
− | | + | |
− | //тест на возможность преобразования
| + | |
− | res_len=MultiByteToWideChar(CP_UTF8,0,utf8_string,-1,0,0);
| + | |
− | if(!res_len)
| + | |
− | {
| + | |
− | return 0;
| + | |
− | }
| + | |
− | | + | |
− | //выделяем память
| + | |
− | pRes = new wchar_t[res_len];
| + | |
− | if(!pRes)
| + | |
− | {
| + | |
− | return 0;
| + | |
− | }
| + | |
− | | + | |
− | //преобразование
| + | |
− | if(!MultiByteToWideChar(CP_UTF8,0,utf8_string,-1,pRes,res_len))
| + | |
− | {
| + | |
− | delete [] pRes;
| + | |
− | return 0;
| + | |
− | }
| + | |
− | | + | |
− | return pRes;
| + | |
− | }
| + | |
− | | + | |
− | //unicode->1251
| + | |
− | static char * unicode_to_1251__dontForgetDeleteArr(const wchar_t *unicode_string)
| + | |
− | {
| + | |
− | char* pRes=0;
| + | |
− | int res_len=0;
| + | |
− | | + | |
− | //тест на возможность преобразования
| + | |
− | res_len = WideCharToMultiByte(1251,0,unicode_string,-1,0,0,0,0);
| + | |
− | if(!res_len)
| + | |
− | {
| + | |
− | return 0;
| + | |
− | }
| + | |
− | | + | |
− | //выделяем память
| + | |
− | pRes=new char[res_len];
| + | |
− | if(!pRes)
| + | |
− | {
| + | |
− | return 0;
| + | |
− | }
| + | |
− | | + | |
− | //преобразование
| + | |
− | if(!WideCharToMultiByte(1251,0,unicode_string,-1,pRes,res_len,0,0))
| + | |
− | {
| + | |
− | delete [] pRes;
| + | |
− | return 0;
| + | |
− | }
| + | |
− | | + | |
− | return pRes;
| + | |
− | }
| + | |
− | | + | |
− | //процедура с примером
| + | |
− | static void Example()
| + | |
− | {
| + | |
− | wchar_t* unicode_string=0;
| + | |
− | char* cp1251_string=0;
| + | |
− | | + | |
− | //исходный текст
| + | |
− | char utf8_string[] = "UTF-8 + СЂСѓСЃСЃРєРёР№ текст";
| + | |
− | | + | |
− | for(;;)
| + | |
− | {
| + | |
− | unicode_string=utf8_to_unicode__dontForgetDeleteArr(utf8_string);
| + | |
− | if(!unicode_string)
| + | |
− | {
| + | |
− | AfxMessageBox("Не удалось конвертировать в unicode!");
| + | |
− | break;
| + | |
− | }
| + | |
− | | + | |
− | cp1251_string = unicode_to_1251__dontForgetDeleteArr(unicode_string);
| + | |
− | | + | |
− | if(!cp1251_string)
| + | |
− | {
| + | |
− | AfxMessageBox("Не удалось конвертировать из unicode!");
| + | |
− | break;
| + | |
− | }
| + | |
− | | + | |
− | break;
| + | |
− | }
| + | |
− | | + | |
− | //cp1251_string - результат
| + | |
− | AfxMessageBox(cp1251_string);
| + | |
− | | + | |
− | //не забываем удалить массивы
| + | |
− | if(unicode_string)
| + | |
− | {
| + | |
− | delete [] unicode_string;
| + | |
− | unicode_string=0;
| + | |
− | }
| + | |
− | | + | |
− | if(cp1251_string)
| + | |
− | {
| + | |
− | delete [] cp1251_string;
| + | |
− | cp1251_string=0;
| + | |
− | }
| + | |
− | }
| + | |
− | };
| + | |
− | | + | |
− | //вызов примера
| + | |
− | coder::Example();
| + | |
− | </syntaxhighlight>
| + | |
− | | + | |
− | ===Как работать с базой данных Access из программы на VC.net?===
| + | |
− | Проще всего подключить ADO компоненты и сделать все на них.
| + | |
− | | + | |
− | Компоненты ADO (ADODC,DATAGRID и т.п.) это ActiveX.
| + | |
− | | + | |
− | В редакторе ресурсов, во всплывающем меню, есть Insert ActiveX Control -> открывается список всех компонент, зарегистрированных на машине. Нужно найти компоненты (Это компоненты от MS и они помечены (OLEDB)) и вставить их в диалог. Для того что бы иметь возможность управлять ими, нужно импортировать обертки для них, это делается при помощи ClassWizard, там есть импорт->класс->указать файл с компонентой (OSX). Появляется список всех классов, объявленных внутри.
| + | |
− | | + | |
− | Расставляем нужные галочки, нажимаем ОК.
| + | |
− | | + | |
− | [[Category:FAQ]]
| + | |