hydrus/hydrus/client/gui/canvas/ClientGUICanvasMedia.py

2008 lines
63 KiB
Python

import fractions
import itertools
import typing
from qtpy import QtCore as QC
from qtpy import QtWidgets as QW
from qtpy import QtGui as QG
from hydrus.core import HydrusConstants as HC
from hydrus.core import HydrusData
from hydrus.core import HydrusGlobals as HG
from hydrus.core import HydrusPaths
from hydrus.client import ClientApplicationCommand as CAC
from hydrus.client import ClientConstants as CC
from hydrus.client import ClientRendering
from hydrus.client.gui import ClientGUIFunctions
from hydrus.client.gui import ClientGUIMedia
from hydrus.client.gui import ClientGUIMediaControls
from hydrus.client.gui import ClientGUIMPV
from hydrus.client.gui import ClientGUIShortcuts
from hydrus.client.gui import QtPorting as QP
from hydrus.client.gui.widgets import ClientGUICommon
from hydrus.client.media import ClientMedia
def ShouldHaveAnimationBar( media, show_action ):
if show_action not in ( CC.MEDIA_VIEWER_ACTION_SHOW_WITH_NATIVE, CC.MEDIA_VIEWER_ACTION_SHOW_WITH_MPV ):
return False
is_animated_image = media.GetMime() in HC.ANIMATIONS
is_audio = media.GetMime() in HC.AUDIO
is_video = media.GetMime() in HC.VIDEO
if show_action == CC.MEDIA_VIEWER_ACTION_SHOW_WITH_MPV:
if ( is_animated_image or is_audio or is_video ) and media.HasDuration():
return True
elif show_action == CC.MEDIA_VIEWER_ACTION_SHOW_WITH_NATIVE:
num_frames = media.GetNumFrames()
has_some_frames = num_frames is not None and num_frames > 1
if ( is_animated_image or is_video ) and has_some_frames:
return True
return False
class Animation( QW.QWidget ):
launchMediaViewer = QC.Signal()
def __init__( self, parent, canvas_type ):
QW.QWidget.__init__( self, parent )
self._canvas_type = canvas_type
# pass up un-button-pressed mouse moves to parent, which wants to do cursor show/hide
self.setMouseTracking( True )
self._media = None
self._left_down_event = None
self._something_valid_has_been_drawn = False
self._playthrough_count = 0
self._num_frames = 1
self._stop_for_slideshow = False
self._current_frame_index = 0
self._current_frame_drawn = False
self._current_timestamp_ms = None
self._next_frame_due_at = HydrusData.GetNowPrecise()
self._slow_frame_score = 1.0
self._paused = True
self._video_container = None
self._canvas_qt_pixmap = None
if self._canvas_type == ClientGUICommon.CANVAS_MEDIA_VIEWER:
shortcut_set = 'media_viewer_media_window'
else:
shortcut_set = 'preview_media_window'
self._my_shortcut_handler = ClientGUIShortcuts.ShortcutsHandler( self, [ shortcut_set ], catch_mouse = True )
def _ClearCanvasBitmap( self ):
if self._canvas_qt_pixmap is not None:
self._canvas_qt_pixmap = None
def _TryToDrawCanvasBitmap( self ):
if self._video_container is None:
size = self.size()
width = size.width()
height = size.height()
self._video_container = ClientRendering.RasterContainerVideo( self._media, ( width, height ), init_position = self._current_frame_index )
if not self._video_container.HasFrame( self._current_frame_index ):
return
my_size = self.size()
my_width = my_size.width()
my_height = my_size.height()
if self._canvas_qt_pixmap is None:
self._canvas_qt_pixmap = HG.client_controller.bitmap_manager.GetQtPixmap( my_width, my_height )
painter = QG.QPainter( self._canvas_qt_pixmap )
current_frame = self._video_container.GetFrame( self._current_frame_index )
( frame_width, frame_height ) = current_frame.GetSize()
scale = my_width / frame_width
painter.setTransform( QG.QTransform().scale( scale, scale ) )
current_frame_image = current_frame.GetQtImage()
painter.drawImage( 0, 0, current_frame_image )
painter.setTransform( QG.QTransform().scale( 1.0, 1.0 ) )
self._current_frame_drawn = True
next_frame_time_s = self._video_container.GetDuration( self._current_frame_index ) / 1000.0
next_frame_ideally_due = self._next_frame_due_at + next_frame_time_s
if HydrusData.TimeHasPassedPrecise( next_frame_ideally_due ):
self._next_frame_due_at = HydrusData.GetNowPrecise() + next_frame_time_s
else:
self._next_frame_due_at = next_frame_ideally_due
self._something_valid_has_been_drawn = True
def _DrawABlankFrame( self, painter ):
new_options = HG.client_controller.new_options
painter.setBackground( QG.QBrush( new_options.GetColour( CC.COLOUR_MEDIA_BACKGROUND ) ) )
painter.eraseRect( painter.viewport() )
self._something_valid_has_been_drawn = True
def ClearMedia( self ):
self.SetMedia( None )
def CurrentFrame( self ):
return self._current_frame_index
def GetAnimationBarStatus( self ):
if self._video_container is None:
buffer_indices = None
else:
buffer_indices = self._video_container.GetBufferIndices()
if self._current_timestamp_ms is None and self._video_container.IsInitialised():
self._current_timestamp_ms = self._video_container.GetTimestampMS( self._current_frame_index )
return ( self._current_frame_index, self._current_timestamp_ms, self._paused, buffer_indices )
def GotoFrame( self, frame_index, pause_afterwards = True ):
if self._video_container is not None and self._video_container.IsInitialised():
if frame_index != self._current_frame_index:
self._current_frame_index = frame_index
self._current_timestamp_ms = None
self._next_frame_due_at = HydrusData.GetNowPrecise()
self._video_container.GetReadyForFrame( self._current_frame_index )
self._current_frame_drawn = False
if pause_afterwards:
self._paused = True
def GotoTimestamp( self, timestamp_ms, round_direction, pause_afterwards = True ):
if self._video_container is not None and self._video_container.IsInitialised():
frame_index = self._video_container.GetFrameIndex( timestamp_ms )
if frame_index == self._current_frame_index:
frame_index += round_direction
if frame_index > self._media.GetNumFrames() - 1:
frame_index = 0
self.GotoFrame( frame_index, pause_afterwards = pause_afterwards )
def HasPlayedOnceThrough( self ):
return self._playthrough_count > 0
def IsPlaying( self ):
return not self._paused
def paintEvent( self, event ):
if not self._current_frame_drawn:
self._TryToDrawCanvasBitmap()
painter = QG.QPainter( self )
if self._canvas_qt_pixmap is None:
self._DrawABlankFrame( painter )
else:
painter.drawPixmap( 0, 0, self._canvas_qt_pixmap )
def Pause( self ):
self._paused = True
def PausePlay( self ):
self._paused = not self._paused
def Play( self ):
self._paused = False
def ProcessApplicationCommand( self, command: CAC.ApplicationCommand ):
command_processed = True
if command.IsSimpleCommand():
action = command.GetSimpleAction()
if action == CAC.SIMPLE_PAUSE_MEDIA:
self.Pause()
elif action == CAC.SIMPLE_PAUSE_PLAY_MEDIA:
self.PausePlay()
elif action == CAC.SIMPLE_MEDIA_SEEK_DELTA:
( direction, duration_ms ) = command.GetSimpleData()
self.SeekDelta( direction, duration_ms )
elif action == CAC.SIMPLE_OPEN_FILE_IN_EXTERNAL_PROGRAM:
if self._media is not None:
self.Pause()
ClientGUIMedia.OpenExternally( self._media )
elif action == CAC.SIMPLE_CLOSE_MEDIA_VIEWER and self._canvas_type == ClientGUICommon.CANVAS_MEDIA_VIEWER:
self.window().close()
elif action == CAC.SIMPLE_LAUNCH_MEDIA_VIEWER and self._canvas_type == ClientGUICommon.CANVAS_PREVIEW:
self.launchMediaViewer.emit()
else:
command_processed = False
else:
command_processed = False
return command_processed
def resizeEvent( self, event ):
size = self.size()
my_width = size.width()
my_height = size.height()
if my_width > 0 and my_height > 0:
if self.size() != event.oldSize():
self._ClearCanvasBitmap()
self._current_frame_drawn = False
self._something_valid_has_been_drawn = False
self.update()
if self._media is not None:
( media_width, media_height ) = self._media.GetResolution()
if self._video_container is not None:
( renderer_width, renderer_height ) = self._video_container.GetSize()
we_just_zoomed_in = my_width > renderer_width or my_height > renderer_height
we_just_zoomed_out = my_width < renderer_width or my_height < renderer_height
if we_just_zoomed_in:
if self._video_container.IsScaled():
target_width = min( media_width, my_width )
target_height = min( media_height, my_height )
self._video_container.Stop()
self._video_container = ClientRendering.RasterContainerVideo( self._media, ( target_width, target_height ), init_position = self._current_frame_index )
elif we_just_zoomed_out:
if my_width < media_width or my_height < media_height: # i.e. new zoom is scaled
self._video_container.Stop()
self._video_container = ClientRendering.RasterContainerVideo( self._media, ( my_width, my_height ), init_position = self._current_frame_index )
def SeekDelta( self, direction, duration_ms ):
if self._video_container is not None and self._video_container.IsInitialised():
new_ts = self._current_timestamp_ms + ( direction * duration_ms )
self.GotoTimestamp( new_ts, direction, pause_afterwards = False )
def StopForSlideshow( self, value ):
self._stop_for_slideshow = value
def SetMedia( self, media, start_paused = False ):
if media == self._media:
return
self._media = media
self._left_down_event = None
self._ClearCanvasBitmap()
self._something_valid_has_been_drawn = False
self._playthrough_count = 0
self._stop_for_slideshow = False
if self._media is not None:
self._num_frames = self._media.GetNumFrames()
else:
self._num_frames = 1
self._current_frame_index = int( ( self._num_frames - 1 ) * HC.options[ 'animation_start_position' ] )
self._current_frame_drawn = False
self._current_timestamp_ms = None
self._next_frame_due_at = HydrusData.GetNowPrecise()
self._slow_frame_score = 1.0
self._paused = start_paused
if self._video_container is not None:
self._video_container.Stop()
self._video_container = None
if self._media is None:
HG.client_controller.gui.UnregisterAnimationUpdateWindow( self )
else:
HG.client_controller.gui.RegisterAnimationUpdateWindow( self )
self.update()
def TIMERAnimationUpdate( self ):
if self._media is None:
return
try:
if self.isVisible():
if self._current_frame_drawn:
if not self._paused and HydrusData.TimeHasPassedPrecise( self._next_frame_due_at ):
num_frames = self._media.GetNumFrames()
next_frame_index = ( self._current_frame_index + 1 ) % num_frames
if next_frame_index == 0:
self._playthrough_count += 1
do_times_to_play_gif_pause = False
if self._media.GetMime() == HC.IMAGE_GIF and not HG.client_controller.new_options.GetBoolean( 'always_loop_gifs' ):
times_to_play_gif = self._video_container.GetTimesToPlayGIF()
# 0 is infinite
if times_to_play_gif != 0 and self._playthrough_count >= times_to_play_gif:
do_times_to_play_gif_pause = True
if self._stop_for_slideshow or do_times_to_play_gif_pause:
self._paused = True
else:
self._current_frame_index = next_frame_index
self._current_timestamp_ms = 0
else:
self._current_frame_index = next_frame_index
if self._current_timestamp_ms is not None and self._video_container is not None and self._video_container.IsInitialised():
duration_ms = self._video_container.GetDuration( self._current_frame_index - 1 )
self._current_timestamp_ms += duration_ms
self._current_frame_drawn = False
if self._video_container is not None:
if not self._current_frame_drawn:
if self._video_container.HasFrame( self._current_frame_index ):
self.update()
except:
HG.client_controller.gui.UnregisterAnimationUpdateWindow( self )
raise
class AnimationBar( QW.QWidget ):
def __init__( self, parent ):
QW.QWidget.__init__( self, parent )
self.setCursor( QG.QCursor( QC.Qt.ArrowCursor ) )
self._media_window = None
self._duration_ms = 1000
self._num_frames = 1
self._last_drawn_info = None
self._currently_in_a_drag = False
self._it_was_playing_before_drag = False
def _DrawBlank( self, painter ):
new_options = HG.client_controller.new_options
painter.setBackground( QG.QBrush( new_options.GetColour( CC.COLOUR_MEDIA_BACKGROUND ) ) )
painter.eraseRect( painter.viewport() )
def _GetAnimationBarStatus( self ):
return self._media_window.GetAnimationBarStatus()
def _GetXFromFrameIndex( self, index, width_offset = 0 ):
if self._num_frames is None or self._num_frames < 2:
return 0
my_width = self.size().width()
return int( ( my_width - width_offset ) * index / ( self._num_frames - 1 ) )
def _GetXFromTimestamp( self, timestamp_ms, width_offset = 0 ):
my_width = self.size().width()
return int( ( my_width - width_offset ) * timestamp_ms / self._duration_ms )
def _CurrentMediaWindowIsBad( self ):
if self._media_window is None:
return True
if not QP.isValid( self._media_window ):
self.ClearMedia()
return True
return False
def _Redraw( self, painter ):
self._last_drawn_info = self._GetAnimationBarStatus()
( current_frame_index, current_timestamp_ms, paused, buffer_indices ) = self._last_drawn_info
my_width = self.size().width()
painter.setPen( QC.Qt.NoPen )
background_colour = QP.GetSystemColour( QG.QPalette.Button )
if paused:
background_colour = ClientGUIFunctions.GetLighterDarkerColour( background_colour )
painter.setBackground( QG.QBrush( background_colour ) )
painter.eraseRect( painter.viewport() )
#
animated_scanbar_height = HG.client_controller.new_options.GetInteger( 'animated_scanbar_height' )
if buffer_indices is not None:
( start_index, rendered_to_index, end_index ) = buffer_indices
if ClientRendering.FrameIndexOutOfRange( rendered_to_index, start_index, end_index ):
rendered_to_index = start_index
start_x = self._GetXFromFrameIndex( start_index )
rendered_to_x = self._GetXFromFrameIndex( rendered_to_index )
end_x = self._GetXFromFrameIndex( end_index )
if start_x != rendered_to_x:
rendered_colour = ClientGUIFunctions.GetDifferentLighterDarkerColour( background_colour )
painter.setBrush( QG.QBrush( rendered_colour ) )
if rendered_to_x > start_x:
painter.drawRect( start_x, 0, rendered_to_x - start_x, animated_scanbar_height )
else:
painter.drawRect( start_x, 0, my_width - start_x, animated_scanbar_height )
painter.drawRect( 0, 0, rendered_to_x, animated_scanbar_height )
if rendered_to_x != end_x:
to_be_rendered_colour = ClientGUIFunctions.GetDifferentLighterDarkerColour( background_colour, 1 )
painter.setBrush( QG.QBrush( to_be_rendered_colour ) )
if end_x > rendered_to_x:
painter.drawRect( rendered_to_x, 0, end_x - rendered_to_x, animated_scanbar_height )
else:
painter.drawRect( rendered_to_x, 0, my_width - rendered_to_x, animated_scanbar_height )
painter.drawRect( 0, 0, end_x, animated_scanbar_height )
painter.setBrush( QG.QBrush( QP.GetSystemColour( QG.QPalette.Shadow ) ) )
animated_scanbar_nub_width = HG.client_controller.new_options.GetInteger( 'animated_scanbar_nub_width' )
num_frames_are_useful = self._num_frames is not None and self._num_frames > 1
nub_x = None
if num_frames_are_useful and current_frame_index is not None:
nub_x = self._GetXFromFrameIndex( current_frame_index, width_offset = animated_scanbar_nub_width )
elif self._duration_ms is not None and current_timestamp_ms is not None:
nub_x = self._GetXFromTimestamp( current_timestamp_ms, width_offset = animated_scanbar_nub_width )
if nub_x is not None:
painter.drawRect( nub_x, 0, animated_scanbar_nub_width, animated_scanbar_height )
#
painter.setPen( QG.QPen() )
progress_strings = []
if num_frames_are_useful:
progress_strings.append( HydrusData.ConvertValueRangeToPrettyString( current_frame_index + 1, self._num_frames ) )
if current_timestamp_ms is not None:
progress_strings.append( HydrusData.ConvertValueRangeToScanbarTimestampsMS( current_timestamp_ms, self._duration_ms ) )
s = ' - '.join( progress_strings )
if len( s ) > 0:
( text_size, s ) = ClientGUIFunctions.GetTextSizeFromPainter( painter, s )
ClientGUIFunctions.DrawText( painter, my_width - text_size.width() - 3, 3, s )
def _ScanToCurrentMousePos( self ):
my_width = self.size().width()
mouse_pos = self.mapFromGlobal( QG.QCursor.pos() )
animated_scanbar_nub_width = HG.client_controller.new_options.GetInteger( 'animated_scanbar_nub_width' )
compensated_x_position = mouse_pos.x() - ( animated_scanbar_nub_width / 2 )
proportion = ( compensated_x_position ) / ( my_width - animated_scanbar_nub_width )
proportion = max( proportion, 0.0 )
proportion = min( 1.0, proportion )
self.update()
if isinstance( self._media_window, Animation ):
current_frame_index = int( proportion * ( self._num_frames - 1 ) + 0.5 )
self._media_window.GotoFrame( current_frame_index )
elif isinstance( self._media_window, ClientGUIMPV.mpvWidget ):
time_index_ms = int( proportion * self._duration_ms )
self._media_window.Seek( time_index_ms )
def ClearMedia( self ):
self._media_window = None
HG.client_controller.gui.UnregisterAnimationUpdateWindow( self )
self.update()
def mouseMoveEvent( self, event ):
if self._CurrentMediaWindowIsBad():
return
CC.CAN_HIDE_MOUSE = False
if self._currently_in_a_drag:
if event.buttons() == QC.Qt.NoButton:
self._currently_in_a_drag = False
return
self._ScanToCurrentMousePos()
def mousePressEvent( self, event ):
if self._CurrentMediaWindowIsBad():
return
CC.CAN_HIDE_MOUSE = False
self._it_was_playing_before_drag = self._media_window.IsPlaying()
if self._it_was_playing_before_drag:
self._media_window.Pause()
self._currently_in_a_drag = True
self._ScanToCurrentMousePos()
def mouseReleaseEvent( self, event ):
CC.CAN_HIDE_MOUSE = True
if self._currently_in_a_drag:
if self._it_was_playing_before_drag:
if not self._CurrentMediaWindowIsBad():
self._media_window.Play()
self._currently_in_a_drag = False
def paintEvent( self, event ):
painter = QG.QPainter( self )
if self._CurrentMediaWindowIsBad():
self._DrawBlank( painter )
else:
self._Redraw( painter )
def SetMediaAndWindow( self, media, media_window ):
self._media_window = media_window
self._duration_ms = max( media.GetDuration(), 1 )
num_frames = media.GetNumFrames()
if num_frames is None:
self._num_frames = num_frames
else:
self._num_frames = max( num_frames, 1 )
self._last_drawn_info = None
self._currently_in_a_drag = False
self._it_was_playing_before_drag = False
HG.client_controller.gui.RegisterAnimationUpdateWindow( self )
self.update()
def TIMERAnimationUpdate( self ):
if self.isVisible():
if not self._media_window or not QP.isValid( self._media_window ):
self.ClearMedia()
return
if self._last_drawn_info != self._GetAnimationBarStatus():
self.update()
class MediaContainer( QW.QWidget ):
launchMediaViewer = QC.Signal()
readyForNeighbourPrefetch = QC.Signal()
def __init__( self, parent, canvas_type, additional_event_filter: QC.QObject ):
QW.QWidget.__init__( self, parent )
self._canvas_type = canvas_type
# If I do not set this, macOS goes 100% CPU endless repaint events!
# My guess is it due to the borked layout
# it means 'I guarantee to cover my whole viewport with pixels, no need for automatic background clear'
self.setAttribute( QC.Qt.WA_OpaquePaintEvent, True )
self.setSizePolicy( QW.QSizePolicy.Fixed, QW.QSizePolicy.Fixed )
self._media = None
self._show_action = CC.MEDIA_VIEWER_ACTION_SHOW_WITH_NATIVE
self._start_paused = False
self._start_with_embed = False
self._media_window = None
self._embed_button = EmbedButton( self )
self._embed_button_widget_event_filter = QP.WidgetEventFilter( self._embed_button )
self._embed_button_widget_event_filter.EVT_LEFT_DOWN( self.EventEmbedButton )
# pass up un-button-pressed mouse moves to parent, which wants to do cursor show/hide
self.setMouseTracking( True )
self._additional_event_filter = additional_event_filter
self._animation_window = Animation( self, self._canvas_type )
self._animation_bar = AnimationBar( self )
self._volume_control = ClientGUIMediaControls.VolumeControl( self, self._canvas_type, direction = 'up' )
self._static_image_window = StaticImage( self, self._canvas_type )
self._static_image_window.readyForNeighbourPrefetch.connect( self.readyForNeighbourPrefetch )
self._volume_control.adjustSize()
self._volume_control.setCursor( QC.Qt.ArrowCursor )
self._animation_window.hide()
self._animation_bar.hide()
self._volume_control.hide()
self._static_image_window.hide()
self._embed_button.hide()
self.hide()
HG.client_controller.sub( self, 'Pause', 'pause_all_media' )
def _DestroyOrHideThisMediaWindow( self, media_window ):
if media_window is not None:
launch_media_viewer_classes = ( Animation, ClientGUIMPV.mpvWidget, StaticImage )
media_window.removeEventFilter( self._additional_event_filter )
if isinstance( media_window, launch_media_viewer_classes ):
try:
media_window.launchMediaViewer.disconnect( self.launchMediaViewer )
except RuntimeError:
pass # lmao, weird 'Failed to disconnect signal launchMediaViewer()' error I couldn't figure out, I guess some out-of-order deleteLater gubbins
if isinstance( media_window, launch_media_viewer_classes ):
media_window.ClearMedia()
media_window.hide()
if isinstance( media_window, ClientGUIMPV.mpvWidget ):
HG.client_controller.gui.ReleaseMPVWidget( media_window )
else:
media_window.deleteLater()
def _HideAnimationBar( self ):
self._animation_bar.ClearMedia()
self._animation_bar.hide()
def _MakeMediaWindow( self ):
old_media_window = self._media_window
destroy_old_media_window = True
do_neighbour_prefetch_emit = True
if self._show_action == CC.MEDIA_VIEWER_ACTION_SHOW_WITH_MPV and not ClientGUIMPV.MPV_IS_AVAILABLE:
self._show_action = CC.MEDIA_VIEWER_ACTION_SHOW_OPEN_EXTERNALLY_BUTTON
HydrusData.ShowText( 'MPV is not available!' )
if self._show_action == CC.MEDIA_VIEWER_ACTION_SHOW_WITH_MPV and self._media.GetMime() == HC.IMAGE_GIF and not self._media.HasDuration():
self._show_action = CC.MEDIA_VIEWER_ACTION_SHOW_WITH_NATIVE
if self._show_action in ( CC.MEDIA_VIEWER_ACTION_DO_NOT_SHOW_ON_ACTIVATION_OPEN_EXTERNALLY, CC.MEDIA_VIEWER_ACTION_DO_NOT_SHOW ):
raise Exception( 'This media should not be shown in the media viewer!' )
elif self._show_action == CC.MEDIA_VIEWER_ACTION_SHOW_OPEN_EXTERNALLY_BUTTON:
self._media_window = OpenExternallyPanel( self, self._media )
elif self._show_action == CC.MEDIA_VIEWER_ACTION_SHOW_WITH_NATIVE:
if self._media.IsStaticImage():
if isinstance( self._media_window, StaticImage ):
destroy_old_media_window = False
self._media_window.hide()
else:
self._media_window = self._static_image_window
self._media_window.SetMedia( self._media )
do_neighbour_prefetch_emit = False
else:
if isinstance( self._media_window, Animation ):
destroy_old_media_window = False
self._media_window.hide()
else:
self._media_window = self._animation_window
self._media_window.SetMedia( self._media, start_paused = self._start_paused )
elif self._show_action == CC.MEDIA_VIEWER_ACTION_SHOW_WITH_MPV:
self._media_window = HG.client_controller.gui.GetMPVWidget( self )
self._media_window.SetCanvasType( self._canvas_type )
self._media_window.SetMedia( self._media, start_paused = self._start_paused )
if ShouldHaveAnimationBar( self._media, self._show_action ):
self._animation_bar.SetMediaAndWindow( self._media, self._media_window )
if isinstance( self._media_window, ClientGUIMPV.mpvWidget ) and self._media.HasAudio():
self._volume_control.show()
else:
self._volume_control.hide()
self._animation_bar.show()
else:
self._HideAnimationBar()
self._volume_control.hide()
media_window_changed = old_media_window != self._media_window
# this has to go after setcanvastype on the mpv window so the filters are in the correct order
if media_window_changed:
self._media_window.installEventFilter( self._additional_event_filter )
launch_media_viewer_classes = ( Animation, ClientGUIMPV.mpvWidget, StaticImage )
if isinstance( self._media_window, launch_media_viewer_classes ):
self._media_window.launchMediaViewer.connect( self.launchMediaViewer )
self._DestroyOrHideThisMediaWindow( old_media_window )
# this forces a flush of the last valid background bmp, so we don't get a flicker of a file from five files ago when we last saw a static image
self.repaint()
if do_neighbour_prefetch_emit:
self.readyForNeighbourPrefetch.emit()
def _SizeAndPositionChildren( self ):
if self._media is not None:
my_size = self.size()
my_width = my_size.width()
my_height = my_size.height()
if self._media_window is None:
self._embed_button.setFixedSize( QC.QSize( my_width, my_height ) )
self._embed_button.move( QC.QPoint( 0, 0 ) )
else:
is_open_externally = isinstance( self._media_window, OpenExternallyPanel )
( media_width, media_height ) = ( my_width, my_height )
if ShouldHaveAnimationBar( self._media, self._show_action ) and not is_open_externally:
animated_scanbar_height = HG.client_controller.new_options.GetInteger( 'animated_scanbar_height' )
media_height -= animated_scanbar_height
if self._volume_control.isVisibleTo( self ):
volume_width = self._volume_control.width()
else:
volume_width = 0
self._animation_bar.setFixedSize( QC.QSize( my_width - volume_width, animated_scanbar_height ) )
self._animation_bar.move( QC.QPoint( 0, my_height - animated_scanbar_height ) )
if self._volume_control.isVisibleTo( self ):
self._volume_control.setFixedSize( QC.QSize( volume_width, animated_scanbar_height ) )
self._volume_control.move( QC.QPoint( self._animation_bar.width(), my_height - animated_scanbar_height ) )
self._media_window.setFixedSize( QC.QSize( media_width, media_height ) )
self._media_window.move( QC.QPoint( 0, 0 ) )
def BeginDrag( self ):
self.parentWidget().BeginDrag()
def ClearMedia( self ):
self._media = None
self._HideAnimationBar()
self._volume_control.hide()
self._DestroyOrHideThisMediaWindow( self._media_window )
self._media_window = None
self.hide()
def EventEmbedButton( self, event ):
self._embed_button.hide()
self._MakeMediaWindow()
self._SizeAndPositionChildren()
def resizeEvent( self, event ):
if self._media is not None:
self._SizeAndPositionChildren()
def GotoPreviousOrNextFrame( self, direction ):
if self._media is not None:
if ShouldHaveAnimationBar( self._media, self._show_action ):
if isinstance( self._media_window, Animation ):
current_frame_index = self._media_window.CurrentFrame()
num_frames = self._media.GetNumFrames()
if direction == 1:
if current_frame_index == num_frames - 1:
current_frame_index = 0
else:
current_frame_index += 1
else:
if current_frame_index == 0:
current_frame_index = num_frames - 1
else:
current_frame_index -= 1
self._media_window.GotoFrame( current_frame_index )
elif isinstance( self._media_window, ClientGUIMPV.mpvWidget ):
self._media_window.GotoPreviousOrNextFrame( direction )
def MouseIsNearAnimationBar( self ):
if self._media is None:
return False
else:
if ShouldHaveAnimationBar( self._media, self._show_action ):
animation_bar_mouse_pos = self._animation_bar.mapFromGlobal( QG.QCursor.pos() )
animation_bar_rect = self._animation_bar.rect()
buffer = 100
test_rect = animation_bar_rect.adjusted( -buffer, -buffer, buffer, buffer )
return test_rect.contains( animation_bar_mouse_pos )
return False
def paintEvent( self, event ):
painter = None
# hackery dackery doo to deal with non-redrawing single-pixel border around the real widget
# we'll fix this when we fix the larger layout/repaint issue
if self._volume_control.isVisible():
painter = QG.QPainter( self )
background_colour = HG.client_controller.new_options.GetColour( CC.COLOUR_MEDIA_BACKGROUND )
painter.setBrush( QG.QBrush( background_colour ) )
painter.setPen( QC.Qt.NoPen )
painter.drawRect( self._volume_control.geometry() )
if self._media_window is not None and self._media_window.isVisible():
return
# this only happens when we are transitioning from one media to another. in the brief period when one media type is going to another, we'll get flicker of the last valid bmp
# mpv embed fun aggravates this
# so instead we do an explicit repaint after the hide and before the new show, to clear our window
if painter is None:
painter = QG.QPainter( self )
background_colour = HG.client_controller.new_options.GetColour( CC.COLOUR_MEDIA_BACKGROUND )
painter.setBrush( QG.QBrush( background_colour ) )
painter.drawRect( painter.viewport() )
def Pause( self ):
if self._media is not None:
if isinstance( self._media_window, ( Animation, ClientGUIMPV.mpvWidget ) ):
self._media_window.Pause()
def PausePlay( self ):
if self._media is not None:
if isinstance( self._media_window, ( Animation, ClientGUIMPV.mpvWidget ) ):
self._media_window.PausePlay()
def ReadyToSlideshow( self ):
if self._media is None:
return False
else:
if isinstance( self._media_window, ( Animation, ClientGUIMPV.mpvWidget ) ):
if not self._media_window.HasPlayedOnceThrough():
return False
if isinstance( self._media_window, StaticImage ):
if not self._media_window.IsRendered():
return False
return True
def SeekDelta( self, direction, duration_ms ):
if self._media is not None:
if isinstance( self._media_window, ( Animation, ClientGUIMPV.mpvWidget ) ):
self._media_window.SeekDelta( direction, duration_ms )
def SetEmbedButton( self ):
self._HideAnimationBar()
self._volume_control.hide()
self._DestroyOrHideThisMediaWindow( self._media_window )
self._media_window = None
self._embed_button.SetMedia( self._media )
self._embed_button.show()
def SetMedia( self, media: ClientMedia.MediaSingleton, initial_size, initial_position, show_action, start_paused, start_with_embed ):
self._media = media
self._show_action = show_action
self._start_paused = start_paused
self._start_with_embed = start_with_embed
if self._start_with_embed:
self.SetEmbedButton()
else:
self._embed_button.hide()
self._MakeMediaWindow()
self.setFixedSize( initial_size )
self.move( initial_position )
self._SizeAndPositionChildren()
if self._media_window is not None:
self._media_window.show()
self.show()
def StopForSlideshow( self, value ):
if isinstance( self._media_window, ( Animation, ClientGUIMPV.mpvWidget ) ):
self._media_window.StopForSlideshow( value )
class EmbedButton( QW.QWidget ):
def __init__( self, parent ):
QW.QWidget.__init__( self, parent )
self._media = None
self._thumbnail_qt_pixmap = None
self.setCursor( QG.QCursor( QC.Qt.PointingHandCursor ) )
HG.client_controller.sub( self, 'update', 'notify_new_colourset' )
def _Redraw( self, painter ):
my_size = self.size()
my_width = my_size.width()
my_height = my_size.height()
center_x = my_width // 2
center_y = my_height // 2
radius = min( 50, center_x, center_y ) - 5
new_options = HG.client_controller.new_options
painter.setBackground( QG.QBrush( new_options.GetColour(CC.COLOUR_MEDIA_BACKGROUND) ) )
painter.eraseRect( painter.viewport() )
if self._thumbnail_qt_pixmap is not None:
scale = my_width / self._thumbnail_qt_pixmap.width()
painter.setTransform( QG.QTransform().scale( scale, scale ) )
painter.drawPixmap( 0, 0, self._thumbnail_qt_pixmap )
painter.setTransform( QG.QTransform().scale( 1.0, 1.0 ) )
painter.setBrush( QG.QBrush( QP.GetSystemColour( QG.QPalette.Button ) ) )
painter.drawEllipse( QC.QPointF( center_x, center_y ), radius, radius )
painter.setBrush( QG.QBrush( QP.GetSystemColour( QG.QPalette.Window ) ) )
# play symbol is a an equilateral triangle
triangle_side = radius * 0.8
half_triangle_side = int( triangle_side // 2 )
cos30 = 0.866
triangle_width = triangle_side * cos30
third_triangle_width = int( triangle_width // 3 )
points = []
points.append( QC.QPoint( center_x - third_triangle_width, center_y - half_triangle_side ) )
points.append( QC.QPoint( center_x + third_triangle_width * 2, center_y ) )
points.append( QC.QPoint( center_x - third_triangle_width, center_y + half_triangle_side ) )
painter.drawPolygon( QG.QPolygon( points ) )
#
painter.setPen( QG.QPen( QP.GetSystemColour( QG.QPalette.Shadow ) ) )
painter.setBrush( QG.QBrush( QG.QColor( QC.Qt.transparent ) ) )
painter.drawRect( 0, 0, my_width, my_height )
def ClearMedia( self ):
self.SetMedia( None )
def paintEvent( self, event ):
painter = QG.QPainter( self )
self._Redraw( painter )
def SetMedia( self, media ):
self._media = media
if self._media is None:
needs_thumb = False
else:
needs_thumb = self._media.GetLocationsManager().IsLocal() and self._media.GetMime() in HC.MIMES_WITH_THUMBNAILS
if needs_thumb:
mime = self._media.GetMime()
thumbnail_path = HG.client_controller.client_files_manager.GetThumbnailPath( self._media )
self._thumbnail_qt_pixmap = ClientRendering.GenerateHydrusBitmap( thumbnail_path, mime ).GetQtPixmap()
self.update()
else:
self._thumbnail_qt_pixmap = None
class OpenExternallyPanel( QW.QWidget ):
def __init__( self, parent, media ):
QW.QWidget.__init__( self, parent )
self._new_options = HG.client_controller.new_options
self._media = media
vbox = QP.VBoxLayout()
if self._media.GetLocationsManager().IsLocal() and self._media.GetMime() in HC.MIMES_WITH_THUMBNAILS:
mime = self._media.GetMime()
thumbnail_path = HG.client_controller.client_files_manager.GetThumbnailPath( self._media )
qt_pixmap = ClientRendering.GenerateHydrusBitmap( thumbnail_path, mime ).GetQtPixmap()
thumbnail_window = ClientGUICommon.BufferedWindowIcon( self, qt_pixmap )
QP.AddToLayout( vbox, thumbnail_window, CC.FLAGS_CENTER )
m_text = HC.mime_string_lookup[ media.GetMime() ]
button = QW.QPushButton( 'open ' + m_text + ' externally', self )
button.setFocusPolicy( QC.Qt.NoFocus )
QP.AddToLayout( vbox, button, CC.FLAGS_EXPAND_BOTH_WAYS )
self.setLayout( vbox )
self.setCursor( QG.QCursor( QC.Qt.PointingHandCursor ) )
button.clicked.connect( self.LaunchFile )
def mousePressEvent( self, event ):
if not ( event.modifiers() & ( QC.Qt.ShiftModifier | QC.Qt.ControlModifier | QC.Qt.AltModifier) ) and event.button() == QC.Qt.LeftButton:
self.LaunchFile()
else:
event.ignore()
def paintEvent( self, event ):
# have to manually repaint background because of parent WA_OpaquePaintEvent
painter = QG.QPainter( self )
background_colour = self._new_options.GetColour( CC.COLOUR_MEDIA_BACKGROUND )
painter.setBackground( QG.QBrush( background_colour ) )
painter.eraseRect( painter.viewport() )
def LaunchFile( self ):
hash = self._media.GetHash()
mime = self._media.GetMime()
client_files_manager = HG.client_controller.client_files_manager
path = client_files_manager.GetFilePath( hash, mime )
launch_path = self._new_options.GetMimeLaunch( mime )
HydrusPaths.LaunchFile( path, launch_path )
class StaticImage( QW.QWidget ):
launchMediaViewer = QC.Signal()
readyForNeighbourPrefetch = QC.Signal()
def __init__( self, parent, canvas_type ):
QW.QWidget.__init__( self, parent )
self._canvas_type = canvas_type
self.setAttribute( QC.Qt.WA_OpaquePaintEvent, True )
# pass up un-button-pressed mouse moves to parent, which wants to do cursor show/hide
self.setMouseTracking( True )
self._media = None
self._image_renderer = None
self._tile_cache = HG.client_controller.GetCache( 'image_tiles' )
self._canvas_tiles = {}
self._is_rendered = False
self._canvas_tile_size = QC.QSize( 768, 768 )
self._zoom = 1.0
if self._canvas_type == ClientGUICommon.CANVAS_MEDIA_VIEWER:
shortcut_set = 'media_viewer_media_window'
else:
shortcut_set = 'preview_media_window'
self._my_shortcut_handler = ClientGUIShortcuts.ShortcutsHandler( self, [ shortcut_set ], catch_mouse = True )
def _ClearCanvasTileCache( self ):
if self._media is None or self.width() == 0 or self.height() == 0:
self._zoom = 1.0
tile_dimension = 0
else:
self._zoom = self.width() / self._media.GetResolution()[ 0 ]
# it is most convenient to have tiles that line up with the current zoom ratio
# 768 is a convenient size for meaty GPU blitting, but as a number it doesn't make for nice multiplication
# a 'nice' size is one that divides nicely by our zoom, so that integer translations between canvas and native res aren't losing too much in the float remainder
# the trick of going ( 123456 // 16 ) * 16 to give you a nice multiple of 16 does not work with floats like 1.4 lmao.
# what we can do instead is phrase 1.4 as 7/5 and use 7 as our int. any number cleanly divisible by 7 is cleanly divisible by 1.4
ideal_tile_dimension = 768
frac = fractions.Fraction( self._zoom ).limit_denominator( 100 )
n = frac.numerator
if n > ideal_tile_dimension:
# we are in extreme zoom land. nice multiples are impossible with reasonable size tiles, so we'll have to settle for some problems
tile_dimension = ideal_tile_dimension
else:
tile_dimension = ( ideal_tile_dimension // n ) * n
tile_dimension = max( min( tile_dimension, 2048 ), 1 )
self._canvas_tile_size = QC.QSize( tile_dimension, tile_dimension )
self._canvas_tiles = {}
self._is_rendered = False
def _DrawBackground( self, painter ):
new_options = HG.client_controller.new_options
painter.setBackground( QG.QBrush( new_options.GetColour( CC.COLOUR_MEDIA_BACKGROUND ) ) )
painter.eraseRect( painter.viewport() )
def _DrawTile( self, tile_coordinate ):
( native_clip_rect, canvas_clip_rect ) = self._GetClipRectsFromTileCoordinates( tile_coordinate )
width = canvas_clip_rect.width()
height = canvas_clip_rect.height()
tile_pixmap = HG.client_controller.bitmap_manager.GetQtPixmap( width, height )
painter = QG.QPainter( tile_pixmap )
self._DrawBackground( painter )
tile = self._tile_cache.GetTile( self._image_renderer, self._media, native_clip_rect, canvas_clip_rect.size() )
painter.drawPixmap( 0, 0, tile.qt_pixmap )
self._canvas_tiles[ tile_coordinate ] = ( tile_pixmap, canvas_clip_rect.topLeft() )
def _GetClipRectsFromTileCoordinates( self, tile_coordinate ) -> typing.Tuple[ QC.QRect, QC.QRect ]:
( tile_x, tile_y ) = tile_coordinate
( my_width, my_height ) = ( self.width(), self.height() )
( normal_canvas_width, normal_canvas_height ) = ( self._canvas_tile_size.width(), self._canvas_tile_size.height() )
( media_width, media_height ) = self._media.GetResolution()
canvas_x = tile_x * self._canvas_tile_size.width()
canvas_y = tile_y * self._canvas_tile_size.height()
canvas_topLeft = QC.QPoint( canvas_x, canvas_y )
canvas_width = normal_canvas_width
if canvas_x + normal_canvas_width > my_width:
# this is the rightmost tile and should be shrunk
canvas_width = my_width % normal_canvas_width
canvas_height = normal_canvas_height
if canvas_y + normal_canvas_height > my_height:
# this is the bottommost tile and should be shrunk
canvas_height = my_height % normal_canvas_height
canvas_width = max( 1, canvas_width )
canvas_height = max( 1, canvas_height )
# if we are the last row/column our size is not this!
canvas_size = QC.QSize( canvas_width, canvas_height )
canvas_clip_rect = QC.QRect( canvas_topLeft, canvas_size )
native_clip_rect = QC.QRect( canvas_topLeft / self._zoom, canvas_size / self._zoom )
# dealing with rounding errors with zoom calc
if native_clip_rect.width() + native_clip_rect.x() > media_width:
native_clip_rect.setWidth( media_width - native_clip_rect.x() )
if native_clip_rect.height() + native_clip_rect.y() > media_height:
native_clip_rect.setHeight( media_height - native_clip_rect.y() )
if native_clip_rect.width() == 0:
native_clip_rect.setWidth( 1 )
if native_clip_rect.height() == 0:
native_clip_rect.setHeight( 1 )
return ( native_clip_rect, canvas_clip_rect )
def _GetTileCoordinateFromPoint( self, pos: QC.QPoint ):
tile_x = pos.x() // self._canvas_tile_size.width()
tile_y = pos.y() // self._canvas_tile_size.height()
return ( tile_x, tile_y )
def _GetTileCoordinatesInView( self, rect: QC.QRect ):
if self.width() == 0 or self.height() == 0 or self._canvas_tile_size.width() == 0 or self._canvas_tile_size.height() == 0:
return []
topLeft_tile_coordinate = self._GetTileCoordinateFromPoint( rect.topLeft() )
bottomRight_tile_coordinate = self._GetTileCoordinateFromPoint( rect.bottomRight() )
i = itertools.product(
range( topLeft_tile_coordinate[0], bottomRight_tile_coordinate[0] + 1 ),
range( topLeft_tile_coordinate[1], bottomRight_tile_coordinate[1] + 1 )
)
return list( i )
def ClearMedia( self ):
self._media = None
self._image_renderer = None
self._ClearCanvasTileCache()
self.update()
def paintEvent( self, event ):
painter = QG.QPainter( self )
if self._image_renderer is None or not self._image_renderer.IsReady():
self._DrawBackground( painter )
return
try:
dirty_tile_coordinates = self._GetTileCoordinatesInView( event.rect() )
for dirty_tile_coordinate in dirty_tile_coordinates:
if dirty_tile_coordinate not in self._canvas_tiles:
self._DrawTile( dirty_tile_coordinate )
for dirty_tile_coordinate in dirty_tile_coordinates:
( tile, pos ) = self._canvas_tiles[ dirty_tile_coordinate ]
painter.drawPixmap( pos, tile )
all_visible_tile_coordinates = self._GetTileCoordinatesInView( self.visibleRegion().boundingRect() )
deletee_tile_coordinates = set( self._canvas_tiles.keys() ).difference( all_visible_tile_coordinates )
for deletee_tile_coordinate in deletee_tile_coordinates:
del self._canvas_tiles[ deletee_tile_coordinate ]
if not self._is_rendered:
self.readyForNeighbourPrefetch.emit()
self._is_rendered = True
except Exception as e:
HydrusData.PrintException( e, do_wait = False )
return
def resizeEvent( self, event ):
self._ClearCanvasTileCache()
def showEvent( self, event ):
self._ClearCanvasTileCache()
def IsRendered( self ):
return self._is_rendered
def ProcessApplicationCommand( self, command: CAC.ApplicationCommand ):
command_processed = True
if command.IsSimpleCommand():
action = command.GetSimpleAction()
if action == CAC.SIMPLE_OPEN_FILE_IN_EXTERNAL_PROGRAM:
if self._media is not None:
ClientGUIMedia.OpenExternally( self._media )
elif action == CAC.SIMPLE_CLOSE_MEDIA_VIEWER and self._canvas_type == ClientGUICommon.CANVAS_MEDIA_VIEWER:
self.window().close()
elif action == CAC.SIMPLE_LAUNCH_MEDIA_VIEWER and self._canvas_type == ClientGUICommon.CANVAS_PREVIEW:
self.launchMediaViewer.emit()
else:
command_processed = False
else:
command_processed = False
return command_processed
def SetMedia( self, media ):
if media == self._media:
return
self._ClearCanvasTileCache()
self._media = media
image_cache = HG.client_controller.GetCache( 'images' )
self._image_renderer = image_cache.GetImageRenderer( self._media )
if not self._image_renderer.IsReady():
HG.client_controller.gui.RegisterAnimationUpdateWindow( self )
self.update()
def TIMERAnimationUpdate( self ):
try:
if self._image_renderer is None or self._image_renderer.IsReady():
self.update()
HG.client_controller.gui.UnregisterAnimationUpdateWindow( self )
except:
HG.client_controller.gui.UnregisterAnimationUpdateWindow( self )
raise