📄 svpaint.cpp
字号:
// Copyright 2007 Google Inc. All Rights Reserved.//// Author: Joern Wanke//// Simple drawing program to illustrate ScrollView capabilities.//// Functionality:// - The menubar is used to select from different sample styles of input.// - With the RMB it is possible to change the RGB values in different// popup menus.// - A LMB click either draws point-to-point, point or text.// - A LMB dragging either draws a line, a rectangle or ellipse.#include "scrollview.h"#include "svmnode.h"#include <stdlib.h>#include <iostream>// The current color values we use, initially white (== ScrollView::WHITE).int rgb[3] = { 255, 255, 255 };class SVPaint : public SVEventHandler { public: SVPaint(const char* server_name);// This is the main event handling function that we need to overwrite, defined// in SVEventHandler. void Notify(const SVEvent* sv_event); private:// The Handler take care of the SVET_POPUP, SVET_MENU, SVET_CLICK and// SVET_SELECTION events. void PopupHandler(const SVEvent* sv_event); void MenuBarHandler(const SVEvent* sv_event); void ClickHandler(const SVEvent* sv_event); void SelectionHandler(const SVEvent* sv_event);// Convenience functions to build little menus. SVMenuNode* BuildPopupMenu(); SVMenuNode* BuildMenuBar();// Our window. ScrollView* window_;// The mode we are in when an SVET_CLICK or an SVET_SELECTION event occurs. int click_mode_; int drag_mode_;// In the point-to-point drawing mode, we need to set a start-point the first// time we call it (e.g. call SetCursor). bool has_start_point_;};// Build a sample popup menu.SVMenuNode* SVPaint::BuildPopupMenu() { SVMenuNode* root = new SVMenuNode(); // Empty root node // Initial color is white, so we all values to 255. root->AddChild("R", // Shown caption. 1, // assoc. command_id. "255", // initial value. "Red Color Value?"); // Shown description. root->AddChild("G", 2, "255", "Green Color Value?"); root->AddChild("B", 3, "255", "Blue Color Value?"); return root;}// Build a sample menu bar.SVMenuNode* SVPaint::BuildMenuBar() { SVMenuNode* root = new SVMenuNode(); // Empty root node // Create some submenus and add them to the root. SVMenuNode* click = root->AddChild("Clicking"); SVMenuNode* drag = root->AddChild("Dragging"); // Put some nodes into the submenus. click->AddChild("Point to Point Drawing", // Caption. 1); // command_id. click->AddChild("Point Drawing", 2); click->AddChild("Text Drawing", 3); drag->AddChild("Line Drawing", 4); drag->AddChild("Rectangle Drawing", 5); drag->AddChild("Ellipse Drawing", 6); return root;}// Takes care of the SVET_POPUP events.// In our case, SVET_POPUP is used to set RGB values.void SVPaint::PopupHandler(const SVEvent* sv_event) { // Since we only have the RGB values as popup items, // we take a shortcut to not bloat up code: rgb[sv_event->command_id - 1] = atoi(sv_event->parameter); window_->Pen(rgb[0], rgb[1], rgb[2]);}// Takes care of the SVET_MENU events.// In our case, we change either the click_mode_ (commands 1-3)// or the drag_mode_ (commands 4-6).void SVPaint::MenuBarHandler(const SVEvent* sv_event) { if ((sv_event->command_id > 0) && (sv_event->command_id < 4)) { click_mode_ = sv_event->command_id; has_start_point_ = false; } else { drag_mode_ = sv_event->command_id; }}// Takes care of the SVET_CLICK events.// Depending on the click_mode_ we are in, either do Point-to-Point drawing,// point drawing, or draw text.void SVPaint::ClickHandler(const SVEvent* sv_event) { switch (click_mode_) { case 1: //Point to Point if (has_start_point_) { window_->DrawTo(sv_event->x, sv_event->y); } else { has_start_point_ = true; window_->SetCursor(sv_event->x, sv_event->y); } break; case 2: //Point Drawing..simulated by drawing a 1 pixel line. window_->Line(sv_event->x, sv_event->y, sv_event->x, sv_event->y); break; case 3: //Text // We show a modal input dialog on our window, then draw the input and // finally delete the input pointer. char* p = window_->ShowInputDialog("Text:"); window_->Text(sv_event->x, sv_event->y, p); delete p; break; }}// Takes care of the SVET_SELECTION events.// Depending on the drag_mode_ we are in, either draw a line, a rectangle or// an ellipse.void SVPaint::SelectionHandler(const SVEvent* sv_event) { switch (drag_mode_) { //FIXME inversed x_size, y_size case 4: //Line window_->Line(sv_event->x, sv_event->y, sv_event->x - sv_event->x_size, sv_event->y - sv_event->y_size); break; case 5: //Rectangle window_->Rectangle(sv_event->x, sv_event->y, sv_event->x - sv_event->x_size, sv_event->y - sv_event->y_size); break; case 6: //Ellipse window_->Ellipse(sv_event->x - sv_event->x_size, sv_event->y - sv_event->y_size, sv_event->x_size, sv_event->y_size); break; }}// The event handling function from ScrollView which we have to overwrite.// We handle CLICK, SELECTION, MENU and POPUP and throw away all other events.void SVPaint::Notify(const SVEvent* sv_event) { if (sv_event->type == SVET_CLICK) { ClickHandler(sv_event); } else if (sv_event->type == SVET_SELECTION) { SelectionHandler(sv_event); } else if (sv_event->type == SVET_MENU) { MenuBarHandler(sv_event); } else if (sv_event->type == SVET_POPUP) { PopupHandler(sv_event); } else {} //throw other events away}// Builds a new window, initializes the variables and event handler and builds// the menu.SVPaint::SVPaint(const char *server_name) { window_ = new ScrollView("ScrollView Paint Example", // window caption 0, 0, // x,y window position 500, 500, // window size 500, 500, // canvas size false, // whether the Y axis is inversed. // this is included due to legacy // reasons for tesseract and enables // us to have (0,0) as the LOWER left // of the coordinate system. server_name); // the server address. // Set the start modes to point-to-point and line drawing. click_mode_ = 1; drag_mode_ = 4; has_start_point_ = false; // Bild our menus and add them to the window. The flag illustrates whether // this is a menu bar. SVMenuNode* popup_menu = BuildPopupMenu(); popup_menu->BuildMenu(window_,false); SVMenuNode* bar_menu = BuildMenuBar(); bar_menu->BuildMenu(window_,true); // Set the initial color values to White (could also be done by // passing (rgb[0], rgb[1], rgb[2]). window_->Pen(ScrollView::WHITE); window_->Brush(ScrollView::WHITE); // Adds the event handler to the window. This actually ensures that Notify // gets called when events occur. window_->AddEventHandler(this); // Set the window visible (calling this is important to actually render // everything. Without this call, the window would also be drawn, but the // menu bars would be missing. window_->SetVisible(true); // Rest this thread until its window is destroyed. // Note that a special eventhandling thread was created when constructing // the window. Due to this, the application will not deadlock here. window_->AwaitEvent(SVET_DESTROY); // We now have 3 Threads running: // (1) The MessageReceiver thread which fetches messages and distributes them // (2) The EventHandler thread which handles all events for window_ // (3) The main thread which waits on window_ for a DESTROY event (blocked)}// If a parameter is given, we try to connect to the given server.// This enables us to test the remote capabilites of ScrollView.int main(int argc, char** argv) { const char* server_name; if (argc > 1) { server_name = argv[1]; } else { server_name = "localhost"; } SVPaint svp(server_name);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -