/* * libjingle * Copyright 2004 Google Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ // Implementation of GtkVideoRenderer #include "talk/media/devices/gtkvideorenderer.h" #include #include #include #include "talk/media/base/videocommon.h" #include "talk/media/base/videoframe.h" namespace cricket { class ScopedGdkLock { public: ScopedGdkLock() { gdk_threads_enter(); } ~ScopedGdkLock() { gdk_threads_leave(); } }; GtkVideoRenderer::GtkVideoRenderer(int x, int y) : window_(NULL), draw_area_(NULL), initial_x_(x), initial_y_(y), width_(0), height_(0) { g_type_init(); // g_thread_init API is deprecated since glib 2.31.0, see release note: // http://mail.gnome.org/archives/gnome-announce-list/2011-October/msg00041.html #if !GLIB_CHECK_VERSION(2, 31, 0) g_thread_init(NULL); #endif gdk_threads_init(); } GtkVideoRenderer::~GtkVideoRenderer() { if (window_) { ScopedGdkLock lock; gtk_widget_destroy(window_); // Run the Gtk main loop to tear down the window. Pump(); } // Don't need to destroy draw_area_ because it is not top-level, so it is // implicitly destroyed by the above. } bool GtkVideoRenderer::SetSize(int width, int height, int reserved) { ScopedGdkLock lock; // If the dimension is the same, no-op. if (width_ == width && height_ == height) { return true; } // For the first frame, initialize the GTK window if ((!window_ && !Initialize(width, height)) || IsClosed()) { return false; } image_.reset(new uint8[width * height * 4]); gtk_widget_set_size_request(draw_area_, width, height); width_ = width; height_ = height; return true; } bool GtkVideoRenderer::RenderFrame(const VideoFrame* video_frame) { if (!video_frame) { return false; } const VideoFrame* frame = video_frame->GetCopyWithRotationApplied(); // Need to set size as the frame might be rotated. if (!SetSize(frame->GetWidth(), frame->GetHeight(), 0)) { return false; } // convert I420 frame to ABGR format, which is accepted by GTK frame->ConvertToRgbBuffer(cricket::FOURCC_ABGR, image_.get(), frame->GetWidth() * frame->GetHeight() * 4, frame->GetWidth() * 4); ScopedGdkLock lock; if (IsClosed()) { return false; } // draw the ABGR image gdk_draw_rgb_32_image(draw_area_->window, draw_area_->style->fg_gc[GTK_STATE_NORMAL], 0, 0, frame->GetWidth(), frame->GetHeight(), GDK_RGB_DITHER_MAX, image_.get(), frame->GetWidth() * 4); // Run the Gtk main loop to refresh the window. Pump(); return true; } bool GtkVideoRenderer::Initialize(int width, int height) { gtk_init(NULL, NULL); window_ = gtk_window_new(GTK_WINDOW_TOPLEVEL); draw_area_ = gtk_drawing_area_new(); if (!window_ || !draw_area_) { return false; } gtk_window_set_position(GTK_WINDOW(window_), GTK_WIN_POS_CENTER); gtk_window_set_title(GTK_WINDOW(window_), "Video Renderer"); gtk_window_set_resizable(GTK_WINDOW(window_), FALSE); gtk_widget_set_size_request(draw_area_, width, height); gtk_container_add(GTK_CONTAINER(window_), draw_area_); gtk_widget_show_all(window_); gtk_window_move(GTK_WINDOW(window_), initial_x_, initial_y_); image_.reset(new uint8[width * height * 4]); return true; } void GtkVideoRenderer::Pump() { while (gtk_events_pending()) { gtk_main_iteration(); } } bool GtkVideoRenderer::IsClosed() const { if (!window_) { // Not initialized yet, so hasn't been closed. return false; } if (!GTK_IS_WINDOW(window_) || !GTK_IS_DRAWING_AREA(draw_area_)) { return true; } return false; } } // namespace cricket