blob: 4e24e4f9d46abcd302e3575cf87ab1253f6c79f4 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <algorithm>
#include "base/macros.h"
#include "base/memory/scoped_ptr.h"
#include "base/message_loop/message_loop.h"
#include "base/strings/string_tokenizer.h"
#include "examples/bitmap_uploader/bitmap_uploader.h"
#include "examples/media_viewer/media_viewer.mojom.h"
#include "mojo/application/application_runner_chromium.h"
#include "mojo/public/c/system/main.h"
#include "mojo/public/cpp/application/application_connection.h"
#include "mojo/public/cpp/application/application_delegate.h"
#include "mojo/public/cpp/application/application_impl.h"
#include "mojo/public/cpp/application/interface_factory_impl.h"
#include "mojo/public/cpp/application/service_provider_impl.h"
#include "mojo/services/public/cpp/view_manager/types.h"
#include "mojo/services/public/cpp/view_manager/view.h"
#include "mojo/services/public/cpp/view_manager/view_manager.h"
#include "mojo/services/public/cpp/view_manager/view_manager_client_factory.h"
#include "mojo/services/public/cpp/view_manager/view_manager_delegate.h"
#include "mojo/services/public/cpp/view_manager/view_observer.h"
#include "mojo/services/public/interfaces/content_handler/content_handler.mojom.h"
#include "skia/ext/platform_canvas.h"
#include "skia/ext/refptr.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/skia/include/core/SkCanvas.h"
#include "third_party/skia/include/core/SkPaint.h"
#include "third_party/skia/include/core/SkScalar.h"
#include "ui/gfx/codec/png_codec.h"
namespace mojo {
namespace examples {
class PNGViewer;
// TODO(aa): Hook up ZoomableMedia interface again.
class PNGView : public ViewManagerDelegate, public ViewObserver {
public:
static void Spawn(URLResponsePtr response,
ServiceProviderImpl* exported_services,
scoped_ptr<ServiceProvider> imported_services,
Shell* shell) {
// PNGView deletes itself when its View is destroyed.
new PNGView(
response.Pass(), exported_services, imported_services.Pass(), shell);
}
private:
static const uint16_t kMaxZoomPercentage = 400;
static const uint16_t kMinZoomPercentage = 20;
static const uint16_t kDefaultZoomPercentage = 100;
static const uint16_t kZoomStep = 20;
PNGView(URLResponsePtr response,
ServiceProviderImpl* exported_services,
scoped_ptr<ServiceProvider> imported_services,
Shell* shell)
: imported_services_(imported_services.Pass()),
shell_(shell),
root_(nullptr),
view_manager_client_factory_(shell, this),
zoom_percentage_(kDefaultZoomPercentage) {
exported_services->AddService(&view_manager_client_factory_);
DecodePNG(response.Pass());
}
virtual ~PNGView() {
if (root_)
root_->RemoveObserver(this);
}
// Overridden from ViewManagerDelegate:
virtual void OnEmbed(ViewManager* view_manager,
View* root,
ServiceProviderImpl* exported_services,
scoped_ptr<ServiceProvider> imported_services) override {
root_ = root;
root_->AddObserver(this);
bitmap_uploader_.reset(new BitmapUploader(root_));
bitmap_uploader_->Init(shell_);
bitmap_uploader_->SetColor(SK_ColorGRAY);
if (!bitmap_.isNull())
DrawBitmap();
}
virtual void OnViewManagerDisconnected(ViewManager* view_manager) override {
// TODO(aa): Need to figure out how shutdown works.
}
// Overridden from ViewObserver:
virtual void OnViewBoundsChanged(View* view,
const Rect& old_bounds,
const Rect& new_bounds) override {
DCHECK_EQ(view, root_);
DrawBitmap();
}
virtual void OnViewDestroyed(View* view) override {
DCHECK_EQ(view, root_);
delete this;
}
void DecodePNG(URLResponsePtr response) {
int content_length = GetContentLength(response->headers);
scoped_ptr<unsigned char[]> data(new unsigned char[content_length]);
unsigned char* buf = data.get();
uint32_t bytes_remaining = content_length;
uint32_t num_bytes = bytes_remaining;
while (bytes_remaining > 0) {
MojoResult result = ReadDataRaw(
response->body.get(), buf, &num_bytes, MOJO_READ_DATA_FLAG_NONE);
if (result == MOJO_RESULT_SHOULD_WAIT) {
Wait(response->body.get(),
MOJO_HANDLE_SIGNAL_READABLE,
MOJO_DEADLINE_INDEFINITE);
} else if (result == MOJO_RESULT_OK) {
buf += num_bytes;
num_bytes = bytes_remaining -= num_bytes;
} else {
break;
}
}
gfx::PNGCodec::Decode(static_cast<const unsigned char*>(data.get()),
content_length,
&bitmap_);
}
void DrawBitmap() {
if (!root_)
return;
skia::RefPtr<SkCanvas> canvas(skia::AdoptRef(skia::CreatePlatformCanvas(
root_->bounds().width, root_->bounds().height, true)));
canvas->drawColor(SK_ColorGRAY);
SkPaint paint;
SkScalar scale =
SkFloatToScalar(zoom_percentage_ * 1.0f / kDefaultZoomPercentage);
canvas->scale(scale, scale);
canvas->drawBitmap(bitmap_, 0, 0, &paint);
bitmap_uploader_->SetBitmap(
skia::GetTopDevice(*canvas)->accessBitmap(true));
}
void ZoomIn() {
if (zoom_percentage_ >= kMaxZoomPercentage)
return;
zoom_percentage_ += kZoomStep;
DrawBitmap();
}
void ZoomOut() {
if (zoom_percentage_ <= kMinZoomPercentage)
return;
zoom_percentage_ -= kZoomStep;
DrawBitmap();
}
void ZoomToActualSize() {
if (zoom_percentage_ == kDefaultZoomPercentage)
return;
zoom_percentage_ = kDefaultZoomPercentage;
DrawBitmap();
}
int GetContentLength(const Array<String>& headers) {
for (size_t i = 0; i < headers.size(); ++i) {
base::StringTokenizer t(headers[i], ": ;=");
while (t.GetNext()) {
if (!t.token_is_delim() && t.token() == "Content-Length") {
while (t.GetNext()) {
if (!t.token_is_delim())
return atoi(t.token().c_str());
}
}
}
}
return 0;
}
SkBitmap bitmap_;
scoped_ptr<ServiceProvider> imported_services_;
Shell* shell_;
View* root_;
ViewManagerClientFactory view_manager_client_factory_;
uint16_t zoom_percentage_;
scoped_ptr<BitmapUploader> bitmap_uploader_;
DISALLOW_COPY_AND_ASSIGN(PNGView);
};
class ContentHandlerImpl : public InterfaceImpl<ContentHandler> {
public:
explicit ContentHandlerImpl(Shell* shell) : shell_(shell) {}
virtual ~ContentHandlerImpl() {}
private:
// Overridden from ContentHandler:
virtual void OnConnect(
const mojo::String& requestor_url,
URLResponsePtr response,
InterfaceRequest<ServiceProvider> service_provider) override {
ServiceProviderImpl* exported_services = new ServiceProviderImpl();
BindToRequest(exported_services, &service_provider);
scoped_ptr<ServiceProvider> remote(
exported_services->CreateRemoteServiceProvider());
PNGView::Spawn(response.Pass(), exported_services, remote.Pass(), shell_);
}
Shell* shell_;
DISALLOW_COPY_AND_ASSIGN(ContentHandlerImpl);
};
class PNGViewer : public ApplicationDelegate {
public:
PNGViewer() {}
private:
// Overridden from ApplicationDelegate:
virtual void Initialize(ApplicationImpl* app) override {
content_handler_factory_.reset(
new InterfaceFactoryImplWithContext<ContentHandlerImpl, Shell>(
app->shell()));
}
// Overridden from ApplicationDelegate:
virtual bool ConfigureIncomingConnection(
ApplicationConnection* connection) override {
connection->AddService(content_handler_factory_.get());
return true;
}
scoped_ptr<InterfaceFactoryImplWithContext<ContentHandlerImpl, Shell> >
content_handler_factory_;
DISALLOW_COPY_AND_ASSIGN(PNGViewer);
};
} // namespace examples
} // namespace mojo
MojoResult MojoMain(MojoHandle shell_handle) {
mojo::ApplicationRunnerChromium runner(new mojo::examples::PNGViewer);
return runner.Run(shell_handle);
}