Во время выполнения я пытаюсь создать для одного столбца пользовательский CListCtrl
(или CMFCListCtrl
, но не CheckListBox
- я хочу иметь возможность добавлять несколько столбцов в будущем), используя MF C , Использование стиля LVS_EX_CHECKBOXES
заставляет все элементы иметь флажок. Нужный элемент управления должен выглядеть следующим образом (у item1 и item3 есть флажки, у item2 нет):
С точки зрения пользователя, желаемый элемент управления списком должен быть создан следующим образом:
int main() {
MyCListCtrl list_control;
list_control.AddItem("item1", true) // true indicates checkbox presence
list_control.AddItem("item2", false) // false - item without checkbox
list_control.AddItem("item3", true) // true indicates checkbox presence
}
До сих пор мне удавалось создать такой элемент управления, но добавление LVS_OWNERDRAWFIXED
вызывает неудачное утверждение при вызове метода базового класса CListCtrl::DrawItem
:
// MyCListCtrl.h
class MyCListCtrl : public CListCtrl {
public:
virtual void DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct) override {
// if the item should be without a checkbox, here I want to move it a few pixels
// to the left so that the checkbox is hidden
...
CListCtrl::DrawItem(lpDrawItemStruct); // call base's DrawItem - without this
// there's no exception but the listbox appears empty
}
};
BOOL MyCDialogEx::OnInitDialog() {
CDialogEx::OnInitDialog();
...
// list being defined somewhere in the header file
list->Create(WS_CHILD | WS_VISIBLE | WS_BORDER | LVS_REPORT | LVS_NOCOLUMNHEADER |
LVS_OWNERDRAWFIXED, // for DrawItem invocation
rect, this, SOME_ID);
list->SetExtendedStyle(list->GetExtendedStyle() | LVS_EX_CHECKBOXES);
// add 1 mandatory column because of LVS_REPORT style
LVCOLUMN lvColumn;
lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH;
lvColumn.fmt = LVCFMT_LEFT;
lvColumn.cx = rect.Width() - 20; // also, how to make this column fit the width exactly?
lvColumn.pszText = nullptr;
list->InsertColumn(0, &lvColumn);
// for now only add 1 testing item and make his checkbox disappear by moving the
// whole item to the left in DrawItem method (called by the system), so that the text
// is aligned to the left list border
list->InsertItem(0, "item1");
...
}
Так выглядит мое (не работающее) решение, если вы знаете, как решить эту проблему, может быть, даже более простым способом, пожалуйста дай мне знать. Спасибо.
РЕДАКТИРОВАТЬ
С помощью @Landstalker, я теперь могу стереть флажок с пользовательским рисунком, но мне все еще нужно переместить текст слева (поэтому он заменяет несуществующий флажок, как на картинке выше). Текущее решение приводит к этому результату:
Это достигается путем обработки сообщения NM_CUSTOMDRAW, например:
void MyCListCtrl::OnCustomDraw(NMHDR* pNMHDR, LRESULT* pResult)
{
*pResult = CDRF_DODEFAULT; // default windows painting
LPNMLVCUSTOMDRAW lpn = (LPNMLVCUSTOMDRAW)pNMHDR;
if (CDDS_PREPAINT == lpn->nmcd.dwDrawStage)
{
*pResult = CDRF_NOTIFYITEMDRAW; // notify on every item
}
else if (CDDS_ITEMPREPAINT == lpn->nmcd.dwDrawStage)
{
int row = lpn->nmcd.dwItemSpec;
if (row == 1) {
lpn->nmcd.rc.left -= 16; // not working
lpn->rcText.left -= 16; // not working
SetItemState(row, INDEXTOSTATEIMAGEMASK(0),
LVIS_STATEIMAGEMASK); // erase checkbox
}
}
}