Logo Search packages:      
Sourcecode: jd version File versions

imagewin.cpp

// ライセンス: GPL2

//#define _DEBUG
#include "jddebug.h"

#include "imageadmin.h"
#include "imagewin.h"

#include "session.h"
#include "dndmanager.h"
#include "command.h"
#include "global.h"


using namespace IMAGE;

#define IMGWIN_FOLDSIZE 10

// タイトルバーの高さの取得方法が分からないのでとりあえずdefineしておく
#define IMGWIN_TITLBARHEIGHT 16


#define FOCUSOUT_TIMEOUT 250 // msec
#define FOCUS_TIME  100 //msec
#define UNGRAB_TIME 100 //msec


// ウィンドウ状態
enum
{
    IMGWIN_NORMAL = 0, // 開いている
    IMGWIN_FOLD,       // 折り畳んでいる
    IMGWIN_EXPANDING,  // 展開中( clock_in() 参照 )
    IMGWIN_UNMAX,      // 最大化 -> 通常
    IMGWIN_UNMAX_FOLD, // 最大化 -> 折り畳み
    IMGWIN_HIDE,       // hide 中
    IMGWIN_UNGRAB      // ブート直後にungrabする( clock_in() 参照 )
};


ImageWin::ImageWin()
    : SKELETON::JDWindow(),
      m_boot( true ),
      m_enable_fold( true ),
      m_transient( false ),
      m_count_focusout( 0 ),
      m_tab( NULL )
{
    // サイズ設定
    m_x = SESSION::get_img_x();
    m_y = SESSION::get_img_y();
    m_width = SESSION::get_img_width();
    m_height = SESSION::get_img_height();

#ifdef _DEBUG
    std::cout << "MessageWin::MessageWin x y w h = " << m_x << " " << m_y << " " << m_width << " " << m_height << std::endl;
#endif

    m_dummywin.resize( 1, 1 );
    m_dummywin.move( -1, -1 );
    m_dummywin.hide();
    m_dummywin.set_skip_taskbar_hint( true );

    set_skip_taskbar_hint( true );
    resize( m_width, 1 );
    move( m_x, m_y );
    focus_out();

    m_vbox_view.pack_end( get_statbar(), Gtk::PACK_SHRINK );

    m_scrwin.set_size_request( 0, 0 );
    m_scrwin.set_policy( Gtk::POLICY_NEVER, Gtk::POLICY_NEVER );
    m_scrwin.add( m_vbox_view );

    pack_remove_end( false, m_scrwin );
    show_all_children();

    property_window_position().set_value( Gtk::WIN_POS_NONE );
    set_transient( true );

    Glib::signal_idle().connect( sigc::mem_fun( *this, &ImageWin::slot_idle ) );

    show_all_children();
}


ImageWin::~ImageWin()
{
#ifdef _DEBUG
    std::cout << "ImageWin::~ImageWin window size : x = " << m_x << " y = " << m_y
              << " w = " << m_width << " h = " << m_height << std::endl;
#endif

    // ウィンドウサイズを保存
    SESSION::set_img_x( m_x );
    SESSION::set_img_y( m_y );
    SESSION::set_img_width( m_width );
    SESSION::set_img_height( m_height );

    SESSION::set_img_shown( false );
    CORE::core_set_command( "restore_focus" );
}


// クロック入力
void ImageWin::clock_in()
{
    int waitcount;

    // 遅延リサイズ( focus_in()にある説明を参照 )
    if( m_mode == IMGWIN_EXPANDING ){

        waitcount = FOCUS_TIME / TIMER_TIMEOUT;
        ++m_counter;
        if( m_counter > waitcount && ! ( m_counter % waitcount ) ){ 

            if( get_height() < m_height ) resize( m_width, m_height );
            else{
#ifdef _DEBUG
                std::cout << "ImageWin::clock_in resize\n";
#endif
                m_mode = IMGWIN_NORMAL;
                SESSION::set_img_shown( true );
                present();
            }
        }
    }

    // ブート直後にフォーカスをメインウィンドウに戻す
    else if( m_mode == IMGWIN_UNGRAB ){

        ++m_counter;
        if( m_counter > UNGRAB_TIME / TIMER_TIMEOUT ){ 
#ifdef _DEBUG
            std::cout << "ImageWin::clock_in ungrab\n";
#endif

            // WMによってはフォーカスが外れない時があるのでlower()して
            // 無理矢理フォーカスを外す
            set_transient( false );
            get_window()->lower();
            set_transient( true );
                
            CORE::core_set_command( "restore_focus", "", "present" );
            m_mode = IMGWIN_FOLD;
        }
    }

    // GNOME環境ではタスクトレイなどで切り替えたときに画像windowがフォーカスされてしまうので
    // メインウィンドウと画像ウィンドウが同時にフォーカスアウトしたら
    // 一時的に transient 指定を外す。メインウィンドウがフォーカスインしたときに
    // ImageAdmin::focus_out() で transient 指定を戻す
    if( SESSION::get_wm() == SESSION::WM_GNOME
        && ! SESSION::is_iconified_win_main() // メインウィンドウが最小化しているときに transient を外すと画像ウィンドウが表示されなくなる
        && ! SESSION::is_focus_win_main() && ! SESSION::is_focus_win_img() ){

        waitcount = FOCUSOUT_TIMEOUT / TIMER_TIMEOUT;

        if( m_count_focusout < waitcount  ) ++m_count_focusout;
        if( m_count_focusout == waitcount ){

#ifdef _DEBUG
            std::cout << "ImageWin::clock_in focus timeout\n";
#endif
            set_transient( false );
            ++m_count_focusout;
        }
    }
    else m_count_focusout = 0;
}


bool ImageWin::slot_idle()
{
    // ブート完了
    if( m_boot ){

#ifdef _DEBUG
        std::cout << "----------------\nImageWin::slot_idle boot end mode = " << m_mode << std::endl;
#endif
        m_boot = false;

        // 遅延させて clock_in()の中でフォーカスをメインウィンドウに戻す
        if( m_mode == IMGWIN_FOLD ){
            m_mode = IMGWIN_UNGRAB;
            m_counter = 0;
        }

        // ImageAdmin 経由で Coreにブートが終わったことを知らせるため
        // ダミーコマンドを送る
        IMAGE::get_admin()->set_command( "imgwin_boot_fin" );
    }

    return false;
}


void ImageWin::set_transient( bool set )
{
    if( set && ! m_transient && CORE::get_mainwindow() ){

#ifdef _DEBUG
    std::cout << "ImageWin::set_transient set = " << set << std::endl;
#endif
        set_transient_for( *CORE::get_mainwindow() );
        m_transient = true;
    }

    // ダミーwindowを使ってtransientを外す
    else if( ! set && m_transient ){

#ifdef _DEBUG
    std::cout << "ImageWin::set_transient set = " << set << std::endl;
#endif
        set_transient_for( m_dummywin );
        m_transient = false;
    }
}



// hide 中
const bool ImageWin::is_hide()
{
    return ( m_mode == IMGWIN_HIDE );
}


//
// ダイアログ表示などでフォーカスが外れてもウインドウを畳まないようにする
//
void ImageWin::set_enable_fold( bool enable )
{
    if( m_enable_fold == enable ) return;

#ifdef _DEBUG
    std::cout << "ImageWin::set_enable_fold " << enable << std::endl;
#endif

    m_enable_fold = enable;

    // XFCE 環境の場合はここでpresent()しておかないとフォーカスが外れる
    if( m_mode == IMGWIN_NORMAL && m_enable_fold ){

        if( SESSION::get_wm() == SESSION::WM_KDE ) CORE::core_set_command( "switch_image" );
        else present();
    }
}



//
// ウィンドウを折り畳んだときの高さ
//
// (TODO) ウィンドウの最小高さの取得方法が分からないのでタブの高さを使っているのをなんとかする
//
int ImageWin::get_min_height()
{
    if( m_tab && m_tab->get_height() > 1 ) return m_tab->get_height() + IMGWIN_TITLBARHEIGHT;

    return 0;
}


void ImageWin::pack_remove_tab( bool unpack, Widget& tab )
{
    m_tab = &tab;
    pack_remove_start( unpack, tab, Gtk::PACK_SHRINK );

}


void ImageWin::pack_remove_view( bool unpack, Widget& view )
{
#ifdef _DEBUG
    std::cout << "ImageWin::pack_remove_view unpack = " << unpack << std::endl;
#endif

    m_vbox_view.pack_remove_end( unpack, view );
    if( ! unpack ) m_vbox_view.show_all();
}


// フォーカスイン
void ImageWin::focus_in()
{
    // メインウィンドウが最小化しているときはメインウィンドウを開く
    if( SESSION::is_iconified_win_main() ){
            m_mode = IMGWIN_UNGRAB;
            m_counter = 0;
            return;
    }

    if( ! m_enable_fold ) return;

    show();
    if( SESSION::is_iconified_win_img() ) deiconify();

    if( ! is_maximized() && get_window() ){
        int x, y;
        get_window()->get_root_origin( x, y );
        if( x != m_x || y != m_y ) move( m_x, m_y );
    }


    // 開く
    //
    // GNOME環境では focus in 動作中に resize() が失敗する時が
    // あるので、遅延させて clock_in() の中でリサイズとpresentする
    if( ! is_maximized() && m_mode != IMGWIN_EXPANDING ){
        m_mode = IMGWIN_EXPANDING;
        m_counter = 0;
    }

#ifdef _DEBUG
    std::cout << "ImageWin::focus_in mode = " << m_mode
              << " maximized = " << is_maximized()
              << " iconified = " << SESSION::is_iconified_win_img()  << std::endl;
#endif
}


// フォーカスアウト
void ImageWin::focus_out()
{
    if( ! m_enable_fold ) return;

    // ポップアップメニューを表示しているかD&D中はfocus_outしない
    if( SESSION::is_popupmenu_shown() ) return;
    if( CORE::get_dnd_manager()->now_dnd() ) return;

    if( is_maximized() ) unmaximize();

    // 折り畳み
    if( m_mode != IMGWIN_FOLD ){
        resize( m_width, IMGWIN_FOLDSIZE );
        m_mode = IMGWIN_FOLD;
        SESSION::set_img_shown( false );
        CORE::core_set_command( "restore_focus" );
    }

#ifdef _DEBUG
    std::cout << "ImageWin::focus_out mode = " << m_mode
              << " maximized = " << is_maximized()
              << " iconified = " << SESSION::is_iconified_win_img() << std::endl;
#endif
}


// フォーカスインイベント
bool ImageWin::on_focus_in_event( GdkEventFocus* event )
{
#ifdef _DEBUG
    std::cout << "ImageWin::on_focus_in_event\n";
#endif

    SESSION::set_focus_win_img( true );

    if( ! m_boot && m_mode != IMGWIN_UNGRAB ) CORE::core_set_command( "switch_image" );

    return Gtk::Window::on_focus_in_event( event );
}


// フォーカスアウトイベント
bool ImageWin::on_focus_out_event( GdkEventFocus* event )
{
#ifdef _DEBUG
    std::cout << "ImageWin::on_focus_out_event\n";
#endif

    SESSION::set_focus_win_img( false );

    if( ! m_boot ) focus_out();

    return Gtk::Window::on_focus_out_event( event );
}


// Xボタンを押した
bool ImageWin::on_delete_event( GdkEventAny* event )
{
#ifdef _DEBUG
    std::cout << "ImageWin::on_delete_event\n";
#endif

    if( is_maximized() ) unmaximize();

    else{

        // hideする前に座標保存
        if( ! is_maximized() && ! SESSION::is_iconified_win_img() && get_window() ){
            get_window()->get_root_origin( m_x, m_y );
        }

        hide();
        m_mode = IMGWIN_HIDE;
        SESSION::set_img_shown( false );
    }

    return true;
}


// 最大、最小化
bool ImageWin::on_window_state_event( GdkEventWindowState* event )
{
    if( ! m_boot ){

        bool maximized = event->new_window_state & GDK_WINDOW_STATE_MAXIMIZED;

        int iconified = event->new_window_state & GDK_WINDOW_STATE_ICONIFIED;

        // 通常 -> アイコン化
        if( ! SESSION::is_iconified_win_img() && iconified ) SESSION::set_img_shown( false );

        // アイコン → 通常
        else if( SESSION::is_iconified_win_img() && ! iconified ) SESSION::set_img_shown( true );

        // 通常 -> 最大化
        if( ! is_maximized() && maximized ){
            m_mode = IMGWIN_NORMAL;
            SESSION::set_img_shown( true );
        }
        else if( is_maximized() && ! maximized ){

            // 最大 -> 折り畳み
            if( m_mode == IMGWIN_FOLD ) m_mode = IMGWIN_UNMAX_FOLD;

            // 最大 -> 通常
            else m_mode = IMGWIN_UNMAX;
        }

#ifdef _DEBUG
        std::cout << "ImageWin::on_window_state_event : "
                  << " mode = " << m_mode
                  << " maximized = " << is_maximized() << " -> " << maximized
                  << " iconified = " << SESSION::is_iconified_win_img() << " -> " << iconified << std::endl;
#endif     

        SESSION::set_iconified_win_img( iconified );
    }

    return SKELETON::JDWindow::on_window_state_event( event );
}


// 移動、サイズ変更イベント
bool ImageWin::on_configure_event( GdkEventConfigure* event )
{
    if( ! m_boot ){

        // 最大 -> 通常に戻る時はリサイズをキャンセル
        if( m_mode == IMGWIN_UNMAX ) m_mode = IMGWIN_NORMAL;
        else if( m_mode == IMGWIN_UNMAX_FOLD ) m_mode = IMGWIN_FOLD;

        // 最大、最小化しているときは除く
        else if( ! is_maximized() && ! SESSION::is_iconified_win_img() ){

            if( get_window() ) get_window()->get_root_origin( m_x, m_y );

            if( m_mode == IMGWIN_FOLD && get_height() > get_min_height() ){
                m_width = get_width();
                m_height = get_height();
            }
        }

#ifdef _DEBUG
        std::cout << "ImageWin::on_configure_event resizing "
                  << " mode = " << m_mode
                  << " show = " << SESSION::is_img_shown()
                  << " maximized = " << is_maximized()
                  << " iconified = " << SESSION::is_iconified_win_img()
                  << " x = " << m_x << " y = " << m_y
                  << " w = " << m_width << " height = " << m_height << std::endl;
#endif     
    }

    return Gtk::Window::on_configure_event( event );
}

Generated by  Doxygen 1.6.0   Back to index