Спасибо Вам за создание действительно нужной рассылки. [Мерси за комплимент ;) – AJ]
Я не так давно начал использовать MFC, и информация, ответы на вопросы мне очень пригодятся. Интересно, что уже в первом прочитанном мной выпуске обсуждался вопрос о курсоре, решение которого я буквально только что искал сам. Поэтому я решил написать этот отзыв в виде нескольких замечаний. На всякий случай оговорю, что все ниже написанное не более чем мое humble opinion :-))
1) Мне не кажется, что описанный Вами универсальный способ изменения курсора нерационален. Не этот обработчик, так CWnd::OnSetCursor все равно вызывается при каждом движении мыши. Поэтому разнице в скорости практически неоткуда взяться. Хотя, LoadCursor действительно лучше вызвать один раз (когда этот курсор впервые устанавливается), сохранить дескриптор нового курсора, который и передавать системной функции SetCursor в обработчике. Мне кажется, это будет по сути то же самое, что делается при обработке события WM_SETCURSOR по умолчанию.
2) О "песочных часах". Если операция, на время которой высвечиваются часики, относительно невелика по времени, и при ее выполнении приложение может не реагировать на другие события, проще всего выделить блок обработки и определить в этом блоке локальный объект CWaitCursor. Все необходимые операции по установке и удалению курсора будут сделаны конструктором и деструктором этого объекта. Именно так, например, в MFC реализованы часики при открытии и сохранении документа. Если же во время операции система реагирует на другие события, то удобнее применять Begin/Restore/EndWaitCursor [Итак, способов уже пять! – AJ]
3) Вы показываете пример с переопределением precreatewindow, в котором регистрируется новый класс окна, и в конце пишете: "В качестве первого параметра для AfxRegisterWndClass можно указать "cs.style", чтобы установить стиль окна по умолчанию." По-моему, это некорректно. Ведь cs.style есть комбинация стилей для окна , а не для класса окна , что требуется при регистрации класса. [Виновен, Ваша честь! – AJ] Мне пришлось столкнуться с этой проблемой, когда я хотел для своего класса CMyView добавить стиль CS_OWNDC, не меняя ничего больше. Дело в том, что при самом первом вызове CMyView::PreCreateWindow класс окна просмотра еще не существует, так как он регистрируется в CView::PreCreateWindow. Поэтому пришлось вызывать родительскую функцию дважды. Вот мое решение, которое, быть может, будет кому-то полезно:
//h-файл
class CMyView :public CView {
public:
virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
protected:
static LPCTSTR lpszViewClassName;
}
//cpp-файл
LPCTSTR CMyView::lpszViewClassName = NULL;
BOOL CMyView::PreCreateWindow(CREATESTRUCT& cs) {
if (lpszViewClassName == NULL) {
CView::PreCreateWindow(cs);
WNDCLASS wc;
//пытаемся получить информацию о классе
//если не удается, выходим с ошибкой
if (!::GetClassInfo(AfxGetInstanceHandle, cs.lpszClass, &wc)) return FALSE;
// теперь изменяем в wc все, что нужно
// например, стиль класса
wc.style |= CS_OWNDC;
// регистрируем класс
lpszViewClassName = AfxRegisterWndClass(wc.style, wc.hCursor, wc.hbrBackground, wc.hIcon);
}
// изменяем класс окна на созданный нами:
cs.lpszClass = lpszViewClassName;
return CView::PreCreateWindow(cs);
}
[Я немножко подправил код и добавил комментарии. Надеюсь, автор на меня не обидится – AJ]
Есть у меня и вопросы, ответы на которые, возможно, будут интересны не только мне.
Q1) В приложении есть операция, которая требует, скажем, больше пяти минут времени, причем по некоторым причинам дальнейшее выполнение не может быть продолжено до завершения этой операции. Хотелось бы, чтобы при этом окно приложения нормально обновлялось, могло быть свернуто-развернуто и т.п. Я нашел некоторое решение, но оно требует создания второго цикла обработки сообщений и потому мне не очень нравится, хотелось бы сделать более естественно.
Q2) Можно ли переопределенный обработчик событий сделать подставляемым (inline)?