Chương Một Hoan nghênh đến với vb6Chương Mười Hai - Dùng Đồ Họa (Phần I)
![]() Ghi chú: Bạn có thể dùng Windows Calculator để hoán chuyển số giữa các dạng Decimal, Binary và Hexadecimal. Chọn View|Scientific thay vì View|Standard. ![]() Function RGBĐể áp dụng Function RGB, ta sẽ viết một chương trình VB6. Bạn hãy khởi động một chương trình VB6 mới, bỏ vào một Label tên Label1 với Caption Red và một Vertical Scroll tên VScroll1. Kế đó select cả hai Label1 và VScroll1 rồi Copy và Paste hai lần để là thêm hai cặp. Đổi Caption của hai Label mới nầy ra Green và Blue. Bây giờ ta có một Array ba Vertical Scrolls cùng tên VScroll1, với index là 0,1 và 2. Đặt một PictureBox tên picColor vào bên phải ba cái VScrolls. Thêm một Label phía dưới, đặt tên nó là lblRGBValue, nhớ clear caption của nó, đừng có để chữ Label1 như dưới đây: ![]() Bây giờ select cả ba VScrolls và edit value của property Max trong cửa sổ Properties thành 255, ý nói khi kéo cái bar của một VScroll1 lên xuống ta giới hạn trị số của nó từ Min là 0 đến Max là 255. ![]() Chuyện chính ta phải làm là viết code để xử lý Event Change của các VScrolls. Vì chúng là một Array nên ta có thể dùng một Sub duy nhất để handle events đến từ cả ba VScrolls. Mỗi lúc một trong 3 VScrolls thay đổi trị số ta sẽ trộn ba màu Red, Green, Blue biểu diễn bởi trị số của 3 VScrolls thành màu BackColor của PictureBox picColor. Đồng thời ta cho hiển thị trị số của ba thành phần màu Red, Green và Blue trong Label lblRGBValue. Bạn hãy double click lên một trong 3 VScrolls rồi viết code như sau:
Bạn hãy khởi động chương trình rồi nắm các bar của 3 VScrolls kéo lên, kéo xuống để xem kết quả. Cửa sổ của chương trình sẽ có dạng giống như dưới đây: ![]() Color MappingNếu dùng Hex Calculator đổi con số 0xFFFFFF ra decimal ta sẽ được 16777215, nếu kể cả số 0 ta sẽ có tổng cộng 16777216 màu. Lúc nãy ta bàn về 8bit (1 byte) và 16bit (2 bytes) color, nhưng ở đây ta nói chuyện 3 byte color. Như thế có thể màn ảnh không đủ khả năng để cung cấp mọi màu mà Function RGB tính ra. Vậy VGA card sẽ làm sao? Thí dụ một cạt VGA chỉ hổ trợ đến 8 bits. Nó sẽ cung cấp 256 màu khác nhau. Nếu Function RGB đói hỏi một màu mà VGA card có thể cung cấp chính xác thì tốt, nếu không nó sẽ tìm cách dùng hai hay ba đóm gần nhau để trộn màu và cho ta ảo tưởng màu ta muốn. Công tác nầy được gọi là Color Mapping và cái màu được làm ra được gọi là custom color. Dùng Intrinsic Color ConstantsMột trong những features của MSWindows là cho ta chọn Color Scheme của Windows theo sở thích. Bình thường, Color Scheme của Windows là Blue, nhưng ta có thể chọn Olive Green hay Silver, nếu ta muốn. ![]() Chỉ khổ nổi nếu ta đã dùng một màu đỏ đậm để hiển thị tuyệt đẹp thứ gì trong chương trình VB6 mà bây giờ user tự nhiên thay đổi Color Scheme thành Olive Green chẳng hạn khiến cho màu đỏ đậm ấy coi chẳng giống ai trong cái Color Scheme mới. Để tránh trường hợp ấy, thay vì nói thẳng ra là màu gì (xanh hay đỏ) ta nói dùng màu vbActiveTitlebar hay vbDesktop, .v.v.
Graphic filesKhi một hình Graphic được lưu trử theo dạng số pixels với màu của chúng như đã nói trên thì ta gọi là một Bit Map và tên file của nó trong disk có extension BMP thí dụ như House.bmp. Lưu trử kiểu nầy cần rất nhiều memory và rất bất tiện để gởi đi hay hiển thị trên một trang Web. Do đó người ta dùng những kỹ thuật để giảm thiểu lượng memory cần để chứa graphic nhưng vẫn giữ được chất lượng của hình ảnh. Có hai dạng Graphic files rất thông dụng trên Web, mang tên với extensions là JPG và GIF. Đặc biệt với GIF files ta có thể chứa cả hoạt họa (animation), tức là một GIF file có thể chứa nhiều hình (gọi là Frames) để chúng lần lượt thay nhau hiển thị, cho người xem có cảm tưởng một vật đang di động. In trên màn ảnhVB6 có method Print cho ta in thẳng trên Form, PictureBox hay Printer. Ba loại control nầy được coi như những khung vải mà họa sĩ vẽ lên. Bạn hãy khởi động một chương trình VB6 mới. Đặt lên form một PictureBox tên Picture1 và một button tên CmdPrintTenLines với Caption Print Ten Lines. DoubleClick lên button nầy và viết code dưới đây:
Bạn hãy chạy thử program rồi click nút Print Ten Lines. Trong trường hợp nầy ta dùng default Font và Color để in 10 hàng. Sau mỗi Print, chương trình tự động xuống hàng. Kế đó, thêm một button tên CmdPrintFontSizes với Caption Print Font Sizes. DoubleClick lên button nầy và viết code dưới đây:
Trong Sub CmdPrintFontSizes_Click, ta thay đổi cở kiểu chữ để cho các con số được in ra lớn lên dần dần và thay đổi màu của các con số bằng cách dùng function QBColor. Để in các con số liên tục không xuống hàng ta dùng method Print với semicolon (;). Bạn hãy chạy chương trình lại. Click nút Print Ten Lines rồi click nút Print Font Sizes, kết quả sẽ giống như dưới đây: ![]() Bây giờ bạn thử minimize cửa sổ của chương trình, kế đó restore nó lại kích thước cũ. Bạn sẽ thấy các hàng ta in lúc nãy không còn trong form hay PictureBox nữa. Lý do là khi ta Print lên form hay PictureBox, các hình ấy được vẽ trong graphic địa phương chớ không được VB6 kể là một phần của cửa sổ. Muốn tránh trở ngại nầy ta phải dặn VB6 nhớ vẽ lại bằng cách set property AutoRedraw của form và Picture1 ra True.
Hệ thống tọa độKhi đặt một Object hay vẽ một cái gì lên màn ảnh (screen) hay form .v.v.. ta cần phải chỉ định Object ấy nằm chỗ nào kể từ (with reference to) cái góc Trên Trái (Top Left) của màn ảnh hay form. Cái góc Trên Trái là Trung tâm tọa độ của screen hay form. Ở đó tọa độ X và Y đều bằng 0, ta viết là 0,0. Nếu ta đi lần qua phải theo chiều rộng của screen thì tọa độ X tăng lên. Nếu ta đi dọc xuống dưới theo chiều cao của screen thì tọa độ của Y tăng lên. Kế đến là đơn vị đo lường ta dùng để biểu diễn khoảng cách. Trong bài trước ta đã nói đến độ mịn của màn ảnh (screen resolution) dựa vào pixel. Ta có thể dùng đơn vị pixel để nói một Object có tọa độ X và Y mỗi chiều bao nhiêu pixels tính từ trung tâm tọa độ. Như thế, ngay cả trên cùng một màn ảnh khi ta tăng độ mịn nó lên thì một Object đã được đặt lên màn ảnh theo đơn vị pixel sẽ xích qua trái và lên trên một ít vì kích thước một pixel bây giờ nhỏ hơn lúc trước một chút. Hình dưới đây minh họa các kích thước của màn ảnh và Form.
Điểm cần biết là có những phần như title bar và border của một form ta không thể vẽ lên được. Do đó diện tích còn lại của form được gọi là Client Area. Chiều rộng và chiều cao của Client Area được gọi là ScaleWidth và ScaleHeight. Nếu muốn khoảng cách từ một Object đến trung tâm tọa độ, hay kích thước của chính Object, không hề thay đổi dù ta có tăng, giảm độ mịn của màn ảnh hay in hình ra printer (thí dụ ta muốn nó luôn luôn dài 5cm chẳng hạn) thì ta dùng hệ thống tọa độ theo đơn vị twips của form. Twips là Default Coordinate System của VB6. Trong hệ thống nầy mỗi điểm là tương đương với 1/567 centimeter. Do đó, nếu bạn vẽ một dường dài 567 twips nó sẽ hiển thị dài 1cm trên màn ảnh, và khi bạn in nó ra, nó cũng dài 1cm trên giấy. Tức là độ dài thật của Object không tùy thuộc vào loại màn ảnh (độ mịn cao hay thấp) hay printer. Người ta nói nó là Device independent coordinate system (Hệ thống tọa độ độc lập với dụng cụ). Nói một cách khác Twips cho ta thật sự what you see is what you get (WYSIWYG - thấy sao có vậy), rất thích hợp với Desktop publishing. Bạn có thể thay đổi hệ thống tọa độ của một form bằng cách edit property ScaleMode qua cửa sổ Properties như sau:
Giống như khi ta Edit Text trong Notepad, Text Cursor (thanh | chớp chớp) là vị trí hiện tại, nơi sẽ hiển thị cái chữ ta đánh sắp tới, trong graphic ta có một Cursor vô hình, nơi sẽ hiển thị cái gì ta sắp Print. Ta chỉ định vị trí của graphic cursor ấy bằng cách cho trị số của CurrentX và CurrentY. Bạn hãy khởi động một dự án VB6 mới và viết code cho Event Resize của form chính như sau:
Thử chạy chương trình và Resize form. Mỗi khi bạn Resize form, chữ X sẽ được dời đến vị trí khoảng chính giữa của Client Area của form. ![]() Dùng GraphicsĐã có một chút căn bản về graphics của VB6, bây giờ ta có thể đặt những graphics lên form. Có hai cách để làm chuyện ấy:
![]() Tùy theo hoàn cảnh, bạn có thể lựa chọn cách nào tiện dụng. PictureBox và ImageDùng PictureBox hay Image là cách dễ nhất để hiển thị một graphic trong form. Lúc thiết kế, bạn có thể đánh thẳng tên của graphic vào property Picture trong cửa sổ Properties. Form cũng nhận property Picture. Bạn cũng có thể click lên bên phải chữ property Picture để browse và chọn một graphic, thường là Bitmap hay Icon. ![]() Sự khác biệt chí giữa Image và PictureBox là Image có property Stretch mà ta có thể set thành True để kéo dãn graphic ra cho chiếm trọn diện tích của Image. Image là một grapgic control lightweight (nhẹ ký), tức là nó không đòi hỏi nhiều memory và chạy nhanh hơn PictureBox. Lý do là PictureBox là một container, tức là nó có thể chứa các controls khác. Ngoài ra, PictureBox cũng cho phép ta vẽ lên trên nó giống như trên form. Trong hình dưới đây, trong lúc thiết kế ta đặt một PictureBox và một Image cùng một cỡ lên cùng một form. Kế đó ta assign cùng một picture hình happy.bmp cho cả hai. Riêng với Image, ta set property Stretch của nó ra True.
Do đó, vì VB6 program chứa luôn graphic chung với nó, ta không cần phải nhắc đến tên của graphic file khi dùng hay deploy, tức là không cần đính kèm tên graphic file trong Setup file cho người ta install. Dưới đây là code mẫu để lúc run-time ta load một graphic tên sad.bmp nằm trong Subfolder tên images của App.path vào Image control tên Image1.
Dĩ nhiên, nếu ta muốn load graphic lúc run-time thì phải cung cấp graphic file riêng. Control ShapeControl Shape cho phép bạn vẽ những hình đơn giản như đường thẳng, hộp, vòng tròn trên form, lúc thiết kế. Sau khi DoubleClick lên control Shape trong Toolbox để thêm một control Shape vào form, bạn chọn loại Shape của nó từ cửa sổ Properties rồi nắm vào một góc của Shape trên form drag lớn nhỏ tùy ý. Muốn sơn bên trong một Shape, bạn chọn màu từ property FillColor. Property FillColor cũng giống như BackColor của các controls khác, nhưng nó chỉ có hiệu lực khi bạn cho property FillStyle một trị số khác hơn là 1- Transparent (trong suốt), thí dụ như 0- Solid (dày đặc). ![]() Control LineTương tự với các properties Fill của Shape, đối với Line bạn có các properties BorderColor, BorderStyle và BorderWidth. Border color chỉ định màu của chính đường thẳng, BorderStyle để cho bạn lựa đường liên tục hay gạch chấm, và BorderWidth để làm cho đường dày to hơn. Các properties nầy cũng áp dụng cho chu vi (đường bao quanh) của các hình chữ nhật, tròn .v.v. ![]() Graphics MethodsTrong khi các Graphical Controls như Shape, Line cho ta vẽ hình lúc thiết kế thì Graphics Methods cho ta vẽ những thứ ấy lúc run-time. Ta cũng có thể chấm từng đóm (pixel) hay copy cả một Picture từ chỗ nầy đến chỗ khác. Chỉ cần một chút kinh nghiệm bạn có thể làm hoạt họa (animation) hay tạo visual effects tuyệt diệu mà không cần phải đụng đến Windows API (Application Programming Interface) để dùng Function BitBlt. Method PaintPictureMethod PaintPicture cho phép bạn copy rất nhanh một khối dữ kiện đồ họa, nói nôm na là một khu vực trong một hình graphic trên form, PictureBox hay Printer đến một nơi khác. Thí dụ bạn copy một hình từ chỗ nầy đến chỗ khác trong form, hay từ form/PictureBox ra Printer Object để một chốc sau bạn in nó ra. Bạn hãy khởi động một dự án VB6 mới và DoubleClick lên PictureBox Icon trong ToolBox để đặt một PictureBox lên form. Đặt tên PictureBox ấy là picGraphic và set property Visible của nó ra False để ta không thấy nó lúc run-time. Bây giờ load một hình vào property Picture của picGraphic bằng cách Browse một Bitmap file từ cửa sổ Properties. Ở đây ta chọn INTL_NO.BMP từ folder \Program Files\Microsoft Visual Studio\Common\Graphics\Bitmaps\Assorted
Trong chương trình nầy ta muốn hễ khi đè nút trái của Mouse xuống và di chuyển Mouse cursor thì khi cursor đi đến đâu, hình INTL_NO được vẽ đến đó. Ta sẽ dùng một Flag để đánh dấu nút-trái-của-Mouse-Down, đặt tên là flgMouseDown. Khi nhận được Event MouseDown ta set flgMouseDown thành True, và khi nhận được Event MouseUp ta reset flgMouseDown thành False. Mỗi lần nhận được Event MouseMove thì nếu flgMouseDown là True ta sẽ PaintPicture INTL_NO. Để xóa background của form, ta thêm một button tên CmdClearForm để chạy graphic method Cls. Dưới đây là liệt kê code mẫu:
![]() Lưu ý là bạn phải declare variable flgMouseDown bên ngoài các Subs để mọi Sub đều thấy và có thể dùng nó. Muốn biết thêm chi tiết về cách dùng method PaintPicture, trong VB6 IDE DoubleClick lên chữ PaintPicture trong code editor để highlight chữ ấy rồi bấm nút F1. Method PSet Ta dùng method PSet (đến từ chữ Point Set) để vẽ một pixel lên form. Ta cần cho biết PSet ở đâu và với màu gì, tức là ta cho nó tọa độ X,Y của pixel và một màu tính từ function RGB. Dưới đây là code để vẽ pixels đủ màu lên form một cách bất chừng (randomly) về vị trí và màu sắc khi user clicks lên form chính:
Trong thí dụ trên ta dùng method Randomize để generate sẵn trong bộ nhớ các con số real bất chừng từ 0 đến 0.999. Sau đó mỗi lần ta gọi Function Rnd(1) là nó sẽ trả về một con số real lấy bất chừng từ bộ số do method Randomize generated. Do đó, Rnd(1) * ScaleWidth sẽ cho ta một con số real có trị số từ 0 đến ScaleWidth. Muốn đổi con số real đó ra Integer, ta dùng Function Int. Khi khởi động chương trình và Click lên form ta sẽ có hình giống như dưới đây: ![]() Mách nước: Để xóa một đóm bạn Pset lại tại chỗ ấy một đóm mới có cùng màu với BackColor của form. Method Line Method Line vẽ một đường thẳng từ một tọa độ nầy đến một tọa độ khác trong màu do ta chỉ định. Với hai methods PSet và Line ta có thể làm được rất nhiều chuyện. Thí dụ muốn cho một vật di động, ta xóa vật ấy bằng cách vẽ lại nó với cùng màu của BackColor của form, rồi vẽ vật ấy ở vị trí mới. Muốn vẽ một đa giác như tam giác hay chữ nhật ta ráp nhiều đường thẳng lại với nhau, đầu của mỗi đường thẳng là cuối của đường thẳng vừa mới được vẽ trước. Muốn sơn Shade bên trong một hình chữ nhật ta dùng PSet..v.v. Có ba cách để chỉ định tọa độ của hai đầu của một đường thẳng ta muốn vẽ:
Line (3000,4000)-(3600,4500), vbMagenta
Line (4000,5100)-(4800,4600), vbGreen Trong thí dụ dưới đây, một hình tam giác được vẽ bằng hai cách coding khác nhau. Khi chạy program để thử, bạn hãy lần lượt click Triangle METHOD I và Triangle METHOD II để thấy cả hai cách vẽ đều y như nhau, chỉ khác màu thôi.
Để vẽ một hình chữ nhật, cách tiện nhất là dùng Step như dưới đây:
Ta còn có thể vẽ một hình chữ nhật với bốn góc tròn như sau:
Ta cũng có thể sơn Shade bên trong hình chữ nhật bằng cách dùng method PSet để chấm các đóm cách nhau chừng 50 pixels như sau:
Muốn Shade đậm hơn, bạn có thể chấm các đóm gần nhau hơn, thí dụ cho cách nhau 30 pixels thay vì 50 pixels. Có một cách khác là tăng trị số của DrawWidth, độ dày của đường vẽ hay đóm. Bây giờ phối hợp cách vẽ hình chữ nhật với method Shade nói trên và method Print ta có thể viết chữ bên trong một khung màu nhạt như sau:
Khi chạy chương trình nầy và click tất cả các buttons trên form, bạn sẽ có hình dưới đây: ![]() Hãy nhớ set property AutoDraw của form ra True để các graphic chương trình vẽ không bị mất khi user minimises form. Bạn cũng có thể dùng những kỹ thuật nói trên với Object Printer để in các mẫu giấy điền chi tiết. Method Circle Ta dùng Method Circle để vẽ hình tròn, hình bầu dục và đường cung, với bên trong trống rỗng hay được sơn đầy bằng một màu ta chỉ định. Ta phải cho biết tọa độ của tâm điểm vòng tròn và bán kính của nó. Bạn hãy khởi động một dự án VB6 mới, đặt lên form một button với tên frmCircle và caption Circle & Lines. DoubleClick lên button ấy và viết code sau đây:
Bây giờ hãy đặt lên form một button khác tên CmdArc và caption Draw Arc. Thay vì vẽ nguyên một vòng tròn, ta sẽ chỉ vẽ một hình vòng cung bằng màu đỏ. Để chỉ định rằng ta sẽ vẽ từ vị trí nào trên vòng tròn đến vị trí nào khác, thí dụ từ 45độ đến 230độ, ta cần phải đổi degree ra đơn vị Radian bằng cách dùng Function Rads như sau:
Vòng cung luôn luôn được vẽ ngược chiều kim đồng hồ. Dưới đây là code để vẽ một đường vòng cung màu đỏ bán kính 800, tâm điểm ở (4000, 2000), từ 45độ đến 230độ:
Ta có thể cho sơn bên trong các hình tròn, hay Pie Slices (một phần của hình tròn) bằng cách set FillStyle bằng 0 và chỉ định màu FillColor. Một Pie Slice là một vòng cung đóng kính bởi hai đường thẳng bán kính ở hai đầu. Muốn vẽ một Pie Slice ta đánh thêm dấu trừ ("-") trước hai trị số Radian, tức là dùng -Rads(45), -Rads(230) thay vì Rads(45), Rads(230). Dưới đây là code vẽ hai Pie Slices, có tâm điểm lệch nhau một tí, đồng thời thêm chú thích 87.5% và 12.5%.
Cách dùng cuối cùng của method Circle là để vẽ một hình bầu dục (Elllipse). Vẽ hình bầu dục giống như vẽ một hình tròn nhưng ta cần cho thêm một parameter gọi là Aspect. Aspect là sự liên hệ giữa bán kính vertical và bán kính horizontal. Thí dụ nếu Aspect=2 thì chiều cao của hình bầu dục gấp đôi chiều ngang, ngược lại, nếu Aspect=0.5 thì chiều ngang sẽ gấp đôi chiều cao. Dưới đây là code ta dùng để vẽ hai hình bầu dục cùng cỡ, một cái màu tím nằm thẳng đứng và một cái màu xanh nằm ngang.
Nếu bạn khởi động chương trình và click cả bốn buttons bạn sẽ thấy hình sau đây: ![]() Property DrawModeThông thường khi ta vẽ, trị số default của property DrawMode là 13- Copy Pen. Có một trị số DrawMode rất thích hợp cho áp dụng hoạt họa là 7- Xor Pen. Muốn xóa một hình vừa vẽ xong ta chỉ cần vẽ lại hình ấy trong DrawMode Xor Pen, không cần biết trước đó background như thế nào, nó sẽ hiện ra trở lại. tải về 1.11 Mb. Chia sẻ với bạn bè của bạn: |