OLD | NEW |
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "remoting/client/plugin/chromoting_plugin.h" | 5 #include "remoting/client/plugin/chromoting_plugin.h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 #include <vector> | 8 #include <vector> |
9 | 9 |
10 #include "base/string_util.h" | 10 #include "base/string_util.h" |
11 #include "base/thread.h" | 11 #include "base/thread.h" |
12 #include "remoting/client/chromoting_client.h" | 12 #include "remoting/client/chromoting_client.h" |
13 #include "remoting/client/host_connection.h" | 13 #include "remoting/client/host_connection.h" |
14 #include "remoting/client/jingle_host_connection.h" | 14 #include "remoting/client/jingle_host_connection.h" |
15 #include "remoting/client/plugin/pepper_view.h" | 15 #include "remoting/client/plugin/pepper_view.h" |
16 #include "remoting/jingle_glue/jingle_thread.h" | 16 #include "remoting/jingle_glue/jingle_thread.h" |
| 17 #include "third_party/ppapi/c/pp_event.h" |
| 18 #include "third_party/ppapi/c/pp_rect.h" |
| 19 #include "third_party/ppapi/cpp/image_data.h" |
17 | 20 |
18 using std::string; | 21 using std::string; |
19 using std::vector; | 22 using std::vector; |
20 | 23 |
21 namespace remoting { | 24 namespace remoting { |
22 | 25 |
23 const char* ChromotingPlugin::kMimeType = | 26 const char* ChromotingPlugin::kMimeType = |
24 "pepper-application/x-chromoting-plugin::Chromoting"; | 27 "pepper-application/x-chromoting-plugin::Chromoting"; |
25 | 28 |
26 ChromotingPlugin::ChromotingPlugin(NPNetscapeFuncs* browser_funcs, | 29 ChromotingPlugin::ChromotingPlugin(PP_Instance instance) |
27 NPP instance) | 30 : pp::Instance(instance), |
28 : PepperPlugin(browser_funcs, instance), width_(0), height_(0), | 31 width_(0), |
29 device_(NULL) { | 32 height_(0) { |
30 } | 33 } |
31 | 34 |
32 ChromotingPlugin::~ChromotingPlugin() { | 35 ChromotingPlugin::~ChromotingPlugin() { |
| 36 if (host_connection_.get()) |
| 37 host_connection_->Disconnect(); |
| 38 |
| 39 // TODO(ajwong): We need to ensure all objects have actually stopped posting |
| 40 // to the message loop before this point. Right now, we don't have a well |
| 41 // defined stop for the plugin process, and the thread shutdown is likely a |
| 42 // race condition. |
| 43 if (network_thread_.get()) |
| 44 network_thread_->Stop(); |
| 45 |
| 46 if (main_thread_.get()) |
| 47 main_thread_->Stop(); |
33 } | 48 } |
34 | 49 |
35 NPError ChromotingPlugin::New(NPMIMEType pluginType, | 50 bool ChromotingPlugin::Init(uint32_t argc, const char* argn[], const char* argv[
]) { |
36 int16 argc, char* argn[], char* argv[]) { | 51 LOG(INFO) << "Started ChromotingPlugin::Init"; |
37 LOG(INFO) << "Started ChromotingPlugin::New"; | |
38 | |
39 // Verify the mime type and subtype | |
40 std::string mime(kMimeType); | |
41 std::string::size_type type_end = mime.find("/"); | |
42 std::string::size_type subtype_end = mime.find(":", type_end); | |
43 if (strncmp(pluginType, kMimeType, subtype_end)) { | |
44 return NPERR_GENERIC_ERROR; | |
45 } | |
46 | 52 |
47 // Extract the URL from the arguments. | 53 // Extract the URL from the arguments. |
48 char* url = NULL; | 54 const char* url = NULL; |
49 for (int i = 0; i < argc; ++i) { | 55 for (uint32_t i = 0; i < argc; ++i) { |
50 if (strcmp(argn[i], "src") == 0) { | 56 if (strcmp(argn[i], "src") == 0) { |
51 url = argv[i]; | 57 url = argv[i]; |
52 break; | 58 break; |
53 } | 59 } |
54 } | 60 } |
55 | 61 |
56 if (!url) { | 62 if (!url) { |
57 return NPERR_GENERIC_ERROR; | 63 return false; |
58 } | 64 } |
59 | 65 |
60 string user_id; | 66 string user_id; |
61 string auth_token; | 67 string auth_token; |
62 string host_jid; | 68 string host_jid; |
63 if (!ParseUrl(url, &user_id, &auth_token, &host_jid)) { | 69 if (!ParseUrl(url, &user_id, &auth_token, &host_jid)) { |
64 LOG(WARNING) << "Could not parse URL: " << url; | 70 LOG(WARNING) << "Could not parse URL: " << url; |
65 return NPERR_GENERIC_ERROR; | 71 return false; |
66 } | 72 } |
67 | 73 |
68 // Setup pepper context. | |
69 device_ = extensions()->acquireDevice(instance(), NPPepper2DDevice); | |
70 | |
71 // Start the threads. | 74 // Start the threads. |
72 main_thread_.reset(new base::Thread("ChromoClientMain")); | 75 main_thread_.reset(new base::Thread("ChromoClientMain")); |
73 if (!main_thread_->Start()) { | 76 if (!main_thread_->Start()) { |
74 LOG(ERROR) << "Main thread failed to start."; | 77 LOG(ERROR) << "Main thread failed to start."; |
75 return NPERR_GENERIC_ERROR; | 78 return false; |
76 } | 79 } |
77 network_thread_.reset(new JingleThread()); | 80 network_thread_.reset(new JingleThread()); |
78 network_thread_->Start(); | 81 network_thread_->Start(); |
79 | 82 |
80 // Create the chromting objects. | 83 // Create the chromting objects. |
81 host_connection_.reset(new JingleHostConnection(network_thread_.get())); | 84 host_connection_.reset(new JingleHostConnection(network_thread_.get())); |
| 85 /* |
82 view_.reset(new PepperView(main_thread_->message_loop(), device_, | 86 view_.reset(new PepperView(main_thread_->message_loop(), device_, |
83 instance())); | 87 instance())); |
84 client_.reset(new ChromotingClient(main_thread_->message_loop(), | 88 */ |
85 host_connection_.get(), view_.get())); | 89 //client_.reset(new ChromotingClient(main_thread_->message_loop(), |
| 90 // host_connection_.get(), view_.get())); |
86 | 91 |
87 // Kick off the connection. | 92 // Kick off the connection. |
88 host_connection_->Connect(user_id, auth_token, host_jid, client_.get()); | 93 //host_connection_->Connect(user_id, auth_token, host_jid, client_.get()); |
89 | 94 |
90 return NPERR_NO_ERROR; | 95 return true; |
91 } | 96 } |
92 | 97 |
93 NPError ChromotingPlugin::Destroy(NPSavedData** save) { | 98 void ChromotingPlugin::ViewChanged(const PP_Rect& position, |
94 host_connection_->Disconnect(); | 99 const PP_Rect& clip) { |
| 100 // TODO(ajwong): This is going to be a race condition when the view changes |
| 101 // and we're in the middle of a Paint(). |
| 102 LOG(INFO) << "ViewChanged " |
| 103 << position.point.x << "," |
| 104 << position.point.y << "," |
| 105 << position.size.width << "," |
| 106 << position.size.height; |
95 | 107 |
96 // TODO(ajwong): We need to ensure all objects have actually stopped posting | 108 // TODO(ajwong): Do we care about the position? Probably not... |
97 // to the message loop before this point. Right now, we don't have a well | 109 if (position.size.width == width_ || position.size.height == height_) |
98 // defined stop for the plugin process, and the thread shutdown is likely a | 110 return; |
99 // race condition. | |
100 network_thread_->Stop(); | |
101 main_thread_->Stop(); | |
102 | 111 |
103 main_thread_.reset(); | 112 width_ = position.size.width; |
104 network_thread_.reset(); | 113 height_ = position.size.height; |
105 return NPERR_NO_ERROR; | 114 |
| 115 device_context_ = pp::DeviceContext2D(width_, height_, false); |
| 116 if (!BindGraphicsDeviceContext(device_context_)) { |
| 117 LOG(ERROR) << "Couldn't bind the device context."; |
| 118 return; |
| 119 } |
| 120 |
| 121 pp::ImageData image(PP_IMAGEDATAFORMAT_BGRA_PREMUL, width_, height_, false); |
| 122 if (!image.is_null()) { |
| 123 for (int y = 0; y < image.height(); y++) { |
| 124 for (int x = 0; x < image.width(); x++) { |
| 125 *image.GetAddr32(x, y) = 0xccff00cc; |
| 126 } |
| 127 } |
| 128 device_context_.ReplaceContents(&image); |
| 129 device_context_.Flush(NULL, this); |
| 130 } else { |
| 131 LOG(ERROR) << "Unable to allocate image."; |
| 132 } |
| 133 |
| 134 //client_->SetViewport(0, 0, width_, height_); |
| 135 //client_->Repaint(); |
106 } | 136 } |
107 | 137 |
108 NPError ChromotingPlugin::SetWindow(NPWindow* window) { | 138 bool ChromotingPlugin::HandleEvent(const PP_Event& event) { |
109 width_ = window->width; | 139 switch (event.type) { |
110 height_ = window->height; | 140 case PP_Event_Type_MouseDown: |
111 | 141 case PP_Event_Type_MouseUp: |
112 client_->SetViewport(0, 0, window->width, window->height); | 142 case PP_Event_Type_MouseMove: |
113 client_->Repaint(); | 143 case PP_Event_Type_MouseEnter: |
114 | 144 case PP_Event_Type_MouseLeave: |
115 return NPERR_NO_ERROR; | |
116 } | |
117 | |
118 int16 ChromotingPlugin::HandleEvent(void* event) { | |
119 NPPepperEvent* npevent = static_cast<NPPepperEvent*>(event); | |
120 | |
121 switch (npevent->type) { | |
122 case NPEventType_MouseDown: | |
123 // Fall through | |
124 case NPEventType_MouseUp: | |
125 // Fall through | |
126 case NPEventType_MouseMove: | |
127 // Fall through | |
128 case NPEventType_MouseEnter: | |
129 // Fall through | |
130 case NPEventType_MouseLeave: | |
131 //client_->handle_mouse_event(npevent); | 145 //client_->handle_mouse_event(npevent); |
132 break; | 146 break; |
133 case NPEventType_MouseWheel: | 147 |
134 case NPEventType_RawKeyDown: | 148 case PP_Event_Type_Char: |
135 break; | |
136 case NPEventType_KeyDown: | |
137 case NPEventType_KeyUp: | |
138 break; | |
139 case NPEventType_Char: | |
140 //client_->handle_char_event(npevent); | 149 //client_->handle_char_event(npevent); |
141 break; | 150 break; |
142 case NPEventType_Minimize: | 151 |
143 case NPEventType_Focus: | 152 default: |
144 case NPEventType_Device: | |
145 break; | 153 break; |
146 } | 154 } |
147 | 155 |
148 return false; | 156 return false; |
149 } | 157 } |
150 | 158 |
151 NPError ChromotingPlugin::GetValue(NPPVariable variable, void* value) { | |
152 return NPERR_NO_ERROR; | |
153 } | |
154 | |
155 NPError ChromotingPlugin::SetValue(NPNVariable variable, void* value) { | |
156 return NPERR_NO_ERROR; | |
157 } | |
158 | |
159 bool ChromotingPlugin::ParseUrl(const std::string& url, | 159 bool ChromotingPlugin::ParseUrl(const std::string& url, |
160 string* user_id, | 160 string* user_id, |
161 string* auth_token, | 161 string* auth_token, |
162 string* host_jid) { | 162 string* host_jid) { |
163 // TODO(ajwong): We should use GURL or something. Don't parse this by hand! | 163 // TODO(ajwong): We should use GURL or something. Don't parse this by hand! |
164 | 164 |
165 // The Url should be of the form: | 165 // The Url should be of the form: |
166 // | 166 // |
167 // chromotocol://<hostid>?user=<userid>&auth=<authtoken>&jid=<hostjid> | 167 // chromotocol://<hostid>?user=<userid>&auth=<authtoken>&jid=<hostjid> |
168 // | 168 // |
(...skipping 18 matching lines...) Expand all Loading... |
187 pos = parts[2].rfind('='); | 187 pos = parts[2].rfind('='); |
188 if (pos == string::npos && (pos + 1) != string::npos) { | 188 if (pos == string::npos && (pos + 1) != string::npos) { |
189 return false; | 189 return false; |
190 } | 190 } |
191 host_jid->assign(parts[2].substr(pos + 1)); | 191 host_jid->assign(parts[2].substr(pos + 1)); |
192 | 192 |
193 return true; | 193 return true; |
194 } | 194 } |
195 | 195 |
196 } // namespace remoting | 196 } // namespace remoting |
OLD | NEW |