| #include "SampleCode.h" |
| #include "SkView.h" |
| #include "SkCanvas.h" |
| #include "SkGPipe.h" |
| #include "SkSockets.h" |
| #include "SkNetPipeController.h" |
| #include "SkCornerPathEffect.h" |
| #include "SkColorPalette.h" |
| #include "SkOSMenu.h" |
| |
| |
| /** |
| * Drawing Client |
| * |
| * A drawing client that allows a user to perform simple brush stokes with |
| * a selected color and brush size. The drawing client communicates with a |
| * drawing server to send/receive data to/from other clients connected to the |
| * same server. The drawing client stores data in fData and fBuffer depending on |
| * the data type. Append type means that the drawing data is a completed stroke |
| * and Replace type means that the drawing data is in progress and will be |
| * replaced by subsequent data. fData and fBuffer are read by a pipe reader and |
| * reproduce the drawing. When the client is in a normal state, the data stored |
| * on the client and the server should be identical. |
| * The drawing client is also able to switch between vector and bitmap drawing. |
| * The drawing client also renders the latest drawing stroke locally in order to |
| * produce better reponses. This can be disabled by calling |
| * controller.disablePlayBack(), which will introduce a lag between the input |
| * and the drawing. |
| * Note: in order to keep up with the drawing data, the client will try to read |
| * a few times each frame in case more than one frame worth of data has been |
| * received and render them together. This behavior can be adjusted by tweaking |
| * MAX_READ_PER_FRAME or disabled by turning fSync to false |
| */ |
| |
| #define MAX_READ_PER_FRAME 5 |
| |
| class DrawingClientView : public SampleView { |
| public: |
| DrawingClientView() { |
| fSocket = NULL; |
| fTotalBytesRead = 0; |
| fPalette = new SkColorPalette; |
| fPalette->setSize(100, 300); |
| fPalette->setVisibleP(true); |
| this->attachChildToFront(fPalette); |
| fPalette->unref(); |
| fBrushSize = 2.5; |
| fAA = false; |
| fPaletteVisible = true; |
| fSync = true; |
| fVector = true; |
| } |
| ~DrawingClientView() { |
| if (fSocket) { |
| delete fSocket; |
| } |
| fData.reset(); |
| fBuffer.reset(); |
| } |
| |
| virtual void requestMenu(SkOSMenu* menu) { |
| menu->setTitle("Drawing Client"); |
| menu->appendTextField("Server IP", "Server IP", this->getSinkID(), |
| "IP address or hostname"); |
| menu->appendSwitch("Vector", "Vector", this->getSinkID(), fVector); |
| menu->appendSlider("Brush Size", "Brush Size", this->getSinkID(), 1.0, |
| 100.0, fBrushSize); |
| menu->appendSwitch("Anti-Aliasing", "AA", this->getSinkID(), fAA); |
| menu->appendSwitch("Show Color Palette", "Palette", this->getSinkID(), |
| fPaletteVisible); |
| menu->appendSwitch("Sync", "Sync", this->getSinkID(), fSync); |
| menu->appendAction("Clear", this->getSinkID()); |
| } |
| |
| protected: |
| |
| static void readData(int cid, const void* data, size_t size, |
| SkSocket::DataType type, void* context) { |
| DrawingClientView* view = (DrawingClientView*)context; |
| view->onRead(cid, data, size, type); |
| } |
| |
| void onRead(int cid, const void* data, size_t size, SkSocket::DataType type) { |
| if (size > 0) { |
| fBuffer.reset(); |
| if (type == SkSocket::kPipeReplace_type) |
| fBuffer.append(size, (const char*)data); |
| else if (type == SkSocket::kPipeAppend_type) |
| fData.append(size, (const char*)data); |
| else { |
| //other types of data |
| } |
| } |
| } |
| |
| bool onQuery(SkEvent* evt) { |
| if (SampleCode::TitleQ(*evt)) { |
| SampleCode::TitleR(evt, "Drawing Client"); |
| return true; |
| } |
| |
| return this->INHERITED::onQuery(evt); |
| } |
| |
| bool onEvent(const SkEvent& evt) {; |
| if (SkOSMenu::FindSliderValue(evt, "Brush Size", &fBrushSize)) |
| return true; |
| |
| SkString s; |
| if (SkOSMenu::FindText(evt, "Server IP", &s)) { |
| if (NULL != fSocket) { |
| delete fSocket; |
| } |
| fSocket = new SkTCPClient(s.c_str(), 40000); |
| fSocket->connectToServer(); |
| fSocket->suspendWrite(); |
| SkDebugf("Connecting to %s\n", s.c_str()); |
| fData.reset(); |
| fBuffer.reset(); |
| this->inval(NULL); |
| return true; |
| } |
| if (SkOSMenu::FindSwitchState(evt, "AA", &fAA) || |
| SkOSMenu::FindSwitchState(evt, "Sync", &fSync)) |
| return true; |
| if (SkOSMenu::FindSwitchState(evt, "Vector", &fVector)) { |
| this->clearBitmap(); |
| return true; |
| } |
| if (SkOSMenu::FindAction(evt, "Clear")) { |
| this->clear(); |
| return true; |
| } |
| if (SkOSMenu::FindSwitchState(evt, "Palette", &fPaletteVisible)) { |
| fPalette->setVisibleP(fPaletteVisible); |
| return true; |
| } |
| return this->INHERITED::onEvent(evt); |
| } |
| |
| virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) { |
| return new Click(this); |
| } |
| |
| virtual bool onClick(SkView::Click* click) { |
| switch (click->fState) { |
| case SkView::Click::kDown_State: |
| fCurrLine.moveTo(click->fCurr); |
| fType = SkSocket::kPipeReplace_type; |
| if (fSocket) |
| fSocket->resumeWrite(); |
| break; |
| case SkView::Click::kMoved_State: |
| fCurrLine.lineTo(click->fCurr); |
| break; |
| case SkView::Click::kUp_State: |
| fType = SkSocket::kPipeAppend_type; |
| break; |
| default: |
| break; |
| } |
| return true; |
| } |
| |
| virtual void onDrawContent(SkCanvas* canvas) { |
| if (fSocket) { |
| if (fSocket->isConnected()) { |
| if (fSync) { |
| int count = 0; |
| while (fSocket->readPacket(readData, this) > 0 && |
| count < MAX_READ_PER_FRAME) |
| ++count; |
| } |
| else |
| fSocket->readPacket(readData, this); |
| } |
| else |
| fSocket->connectToServer(); |
| } |
| size_t bytesRead = 0; |
| SkGPipeReader::Status status; |
| SkCanvas bufferCanvas(fBase); |
| SkCanvas* tempCanvas; |
| while (fTotalBytesRead < fData.count()) { |
| if (fVector) |
| tempCanvas = canvas; |
| else |
| tempCanvas = &bufferCanvas; |
| SkGPipeReader reader(tempCanvas); |
| status = reader.playback(fData.begin() + fTotalBytesRead, |
| fData.count() - fTotalBytesRead, |
| &bytesRead); |
| SkASSERT(SkGPipeReader::kError_Status != status); |
| fTotalBytesRead += bytesRead; |
| } |
| if (fVector) |
| fTotalBytesRead = 0; |
| else |
| canvas->drawBitmap(fBase, 0, 0, NULL); |
| |
| size_t totalBytesRead = 0; |
| while (totalBytesRead < fBuffer.count()) { |
| SkGPipeReader reader(canvas); |
| status = reader.playback(fBuffer.begin() + totalBytesRead, |
| fBuffer.count() - totalBytesRead, |
| &bytesRead); |
| SkASSERT(SkGPipeReader::kError_Status != status); |
| totalBytesRead += bytesRead; |
| } |
| |
| SkNetPipeController controller(canvas); |
| SkGPipeWriter writer; |
| SkCanvas* writerCanvas = writer.startRecording(&controller, |
| SkGPipeWriter::kCrossProcess_Flag); |
| |
| //controller.disablePlayback(); |
| SkPaint p; |
| p.setColor(fPalette->getColor()); |
| p.setStyle(SkPaint::kStroke_Style); |
| p.setStrokeWidth(fBrushSize); |
| p.setStrokeCap(SkPaint::kRound_Cap); |
| p.setStrokeJoin(SkPaint::kRound_Join); |
| p.setAntiAlias(fAA); |
| p.setPathEffect(new SkCornerPathEffect(55))->unref(); |
| writerCanvas->drawPath(fCurrLine, p); |
| writer.endRecording(); |
| |
| controller.writeToSocket(fSocket, fType); |
| if (fType == SkSocket::kPipeAppend_type && fSocket) { |
| fSocket->suspendWrite(); |
| fCurrLine.reset(); |
| } |
| |
| this->inval(NULL); |
| } |
| |
| virtual void onSizeChange() { |
| this->INHERITED::onSizeChange(); |
| fPalette->setLoc(this->width()-100, 0); |
| fBase.setConfig(SkBitmap::kARGB_8888_Config, this->width(), this->height()); |
| fBase.allocPixels(NULL); |
| this->clearBitmap(); |
| } |
| |
| private: |
| void clear() { |
| fData.reset(); |
| fBuffer.reset(); |
| fCurrLine.reset(); |
| fTotalBytesRead = 0; |
| this->clearBitmap(); |
| } |
| void clearBitmap() { |
| fTotalBytesRead = 0; |
| fBase.eraseColor(fBGColor); |
| } |
| SkTDArray<char> fData; |
| SkTDArray<char> fBuffer; |
| SkBitmap fBase; |
| SkPath fCurrLine; |
| SkTCPClient* fSocket; |
| SkSocket::DataType fType; |
| SkColorPalette* fPalette; |
| bool fPaletteVisible; |
| size_t fTotalBytesRead; |
| SkScalar fBrushSize; |
| bool fAA; |
| bool fSync; |
| bool fVector; |
| |
| typedef SampleView INHERITED; |
| }; |
| |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| static SkView* MyFactory() { return new DrawingClientView; } |
| static SkViewRegister reg(MyFactory); |