Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Visual1.doc
Скачиваний:
8
Добавлен:
07.03.2016
Размер:
4.35 Mб
Скачать

5.9. Копіювання бітових образів

Приклад копіювання бітових образів знаходиться у папці DISK\GDI\GDI7.

У наступному прикладі продемонструємо роботу зі створення та копіювання бітових зображень.

Створимо програму з назвою DDB із MDI інтерфейсом. Принцип роботи програми буде таким:

1) при включенні у вікні виду будуть малюватись 4 прямокутника;

2) при подвійному натисненні кнопкою миші на один із них буде створюватись копія обраної прямокутної області у пам'яті (ця копія контексту екрану вікна буде робитись з використанням класу CBitmap);

3) при подвійному натисненні кнопкою миші не на прямокутну область будемо виводити у вікні раніше скопійовану область екрану, якщо вона буде повністтю вміщатися у вікно;

4) при зміненні розмірів вікна прямокутники зміняться і тому обрана область екрану буде не доступна для виведення на екран.

Додамо у клас виду наступні змінні:

class CDDBAppView : public CView

{

.............................................................

public:

bool was_copied;

BYTE* m_Buffer;

CRect m_Rect;

.............................................................

};

Змінна was_copied буде вказувати на те чи відбулось копіювання прямокутної області, тобто чи користувач натискав на якийсь із прямокутників чи ні. У m_Buffer буде міститись буфер із обраною областю екрану. Призначення прямокутника m_Rect розглянемо далі.

Основна робота програми буде виконуватись у двох функціях це OnDraw і OnLButtonDblClk. Перша функція створюється разом із програмою а другу функцію додамо через майстер. У функції OnDraw будемо виводити прямокутники наступним чином:

void CDDBAppView::OnDraw(CDC* pDC)

{

CDDBAppDoc* pDoc = GetDocument();

ASSERT_VALID(pDoc);

CRect rect;

int wd;

GetClientRect(&rect);

DWORD col[4] =

{

RGB( 0, 0, 0 ),

RGB( 255, 0, 0 ),

RGB( 0, 255, 0 ),

RGB( 0, 0, 255 ),

};

rect.right = (rect.right - 10)/4*4;

m_Rect = CRect( 10, 10, rect.right,10 + height);

wd = m_Rect.Width()/4;

rect = m_Rect;

rect.right = rect.left + wd;

for(int i = 0; i < 4; i++)

{

pDC->FillRect(rect,&CBrush(col[i]));

rect.OffsetRect(wd,0);

}

was_copied = false;

}

У прямокутнику m_Rect будуть зберігатись координати області із усіма прямокутниками. Локальна змінна rect потрібна для виведення на екран окремих прямокутників.

Функція GetClientRect визначає розміри клієнтського вікна. На рис.5.17. приведено результат виведення прямокутників.

Рис. 5.17. Виведення прямокутників у контексті вікна

Функція OnLButtonDblClk котра реагуватиме на подвійне клацання мишею має такий вид:

void CDDBAppView::OnLButtonDblClk(UINT nFlags, CPoint point)

{

CClientDC pDC( this );

CDC memDC;

CBitmap m_Bitmap;

int size;

int wd = m_Rect.Width()/4;

m_Bitmap.CreateCompatibleBitmap

( &pDC, wd, m_Rect.Height() );

BITMAP bm;

m_Bitmap.GetBitmap(&bm);

// size - розмір

// bm.bmWidth - ширина бітового образу

// bm.bmHeight - висота бітового образу

// bm.bmBitsPixel - кількість біт на піксель;

size = bm.bmWidth*bm.bmHeight*bm.bmBitsPixel/8;

if(!m_Buffer)

m_Buffer = new BYTE[size];

memDC.CreateCompatibleDC( &pDC );

CBitmap* oldBitmap = memDC.SelectObject( &m_Bitmap );

if( m_Rect.PtInRect( point ) )

{

int nRect = (point.x - m_Rect.left)/wd;

memDC.BitBlt(0, 0, wd, height, &pDC,

m_Rect.left + nRect*wd, 10, SRCCOPY);

m_Bitmap.GetBitmapBits(size, m_Buffer);

was_copied = true;

}

else

{

CRect rect;

GetClientRect( rect );

if( was_copied && (point.x < (rect.right - wd)) &&

(point.y < (rect.bottom - height)) )

{

m_Bitmap.SetBitmapBits(size, m_Buffer );

pDC.BitBlt(point.x, point.y, wd, height, &memDC, 0,

0, SRCCOPY);

}

}

memDC.SelectObject( &oldBitmap );

CView::OnLButtonDblClk(nFlags, point);

}

Спочатку функції ми підготовлюємо об’єкти: pDC, memDC, m_Bitmap і bm. Змінна wd буде містити ширину прямокутника у пікселях. Змінна size потрібна для обчислення кількості байтів для зберігання бітового зображення. Кількість байтів у бітовому образі обчислюється за формулою: ширина образу * висота образу * кількість біт у пікселі / 8. Далі створюємо контекст у пам’яті memDC, за допомогою функції CreateCompatibleDC.

Функція PtInRect із класу CRect перевіряє чи потрапляє точка у прямокутник. У прямокутнику m_Rect знаходиться область, яку займають прямокутники. Отже в умові if відбувається перевірка чи потрапили ми при натисненні у будь-який із цих прямокутників. Якщо потрапили у область m_Rect, тоді обчислюємо у змінній nRect номер прямокутника, на якому ми натиснули.

Механізм роботи з виведенням бітових зображень полягає у наявності двох контекстів: контексту для малювання pDC та контексту пам’яті memDC. Так, зображення можна виводити з одного контексту в інший і навпаки.

Для того щоб вивести зображення в контекст пам’яті memDC з контексту малювання pDC скористаємось функцією BitBlt у наступному рядку коду.

memDC.BitBlt(0, 0, wd, height, &pDC,

m_Rect.left + nRect*wd, 10, SRCCOPY);

За допомогою цього рядку у контекст пам’яті записується бітовий образ прямокутника з номером nRect. Бітовий образ фактично записується до об’єкту m_Bitmap класу CBitmap, так як у функції SelectObject завантажується саме цей бітовий образ. Після запису в бітовий образ прямокутнику за допомогою GetBitmapBits класу CBitmap відбувається запис кольорів пікселів до масиву m_Buffer. При цьому змінній was_copied привласнюється значення true, що говорить про те, що бітовий образ з вибраним прямокутником знаходиться в пам’яті.

Для того щоб вивести зображення на екран дисплею з пам’яті, його спочатку треба завантажити до контексту memDC, а потім, використовуючи функцію BitBlt, вивести його на екран. Це відбувається наступним чином.

m_Bitmap.SetBitmapBits(size, m_Buffer );

pDC.BitBlt(point.x, point.y, wd, height, &memDC, 0,

0, SRCCOPY);

Спочатку в об’єкт m_Bitmap записується масив кольорів пікселів бітового образу за допомогою функції SetBitmapBits, а потім за допомогою функції BitBlt прямокутник виводиться на екран дисплею у заданому місці.

При зміні розміру вікна (функція OnSize) прямокутники будуть перемальовані, отже треба зробити видалення масиву кольорів пікселів m_Buffer та змінній was_copied привласнити значення false. Наведемо текст даної функції.

void CDDBAppView::OnSize(UINT nType, int cx, int cy)

{

if(m_Buffer)

{

delete []m_Buffer;

m_Buffer = 0;

}

was_copied = false;

CView::OnSize(nType, cx, cy);

}

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]