You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
242 lines
12 KiB
242 lines
12 KiB
#! /usr/bin/env python3
|
|
# -*- coding: utf-8 -*-
|
|
"""
|
|
Copyright 2017, Nils Hilbricht, Germany ( https://www.hilbricht.net )
|
|
|
|
This file is part of Laborejo ( https://www.laborejo.org )
|
|
|
|
Laborejo is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
"""
|
|
|
|
from PyQt5 import QtCore, QtGui, QtWidgets, QtWidgets
|
|
from .constantsAndConfigs import constantsAndConfigs
|
|
import engine.api as api
|
|
|
|
pen = QtGui.QPen()
|
|
pen.setCapStyle(QtCore.Qt.RoundCap)
|
|
pen.setJoinStyle(QtCore.Qt.RoundJoin)
|
|
pen.setWidth(2)
|
|
#pen.setColor(QtGui.QColor("red"))
|
|
|
|
class PitchCursor(QtWidgets.QGraphicsRectItem):
|
|
def __init__(self):
|
|
"""Does not need the actual dotOnLine.
|
|
this is done by the parent cursor.
|
|
This is just a fancy rect."""
|
|
#super().__init__(constantsAndConfigs.magicPixel, 2, constantsAndConfigs.magicPixel*3, constantsAndConfigs.stafflineGap/3)
|
|
super().__init__(0, 1, constantsAndConfigs.magicPixel*3, 5)
|
|
self.setBrush(QtGui.QColor("cyan"))
|
|
self.setPen(pen)
|
|
self.setEnabled(False)
|
|
|
|
class PositionCursor(QtWidgets.QGraphicsRectItem):
|
|
def __init__(self):
|
|
"""Does not need the actual position.
|
|
this is done by the parent cursor
|
|
This is just a fancy rect."""
|
|
super().__init__(-1.5*constantsAndConfigs.magicPixel, -3.5*constantsAndConfigs.stafflineGap, constantsAndConfigs.magicPixel*1.5, 7*constantsAndConfigs.stafflineGap)
|
|
self.setBrush(QtGui.QColor("cyan"))
|
|
self.setPen(pen)
|
|
self.setScale(0.8)
|
|
self.setEnabled(False)
|
|
|
|
class Cursor(QtWidgets.QGraphicsItemGroup):
|
|
"""A cursor that shows the vertical
|
|
as well as the horizontal position
|
|
"""
|
|
def __init__(self):
|
|
super().__init__()
|
|
self.cursorExportObject = None #the last cached cursor object aka the current cursor position
|
|
self.pitch = PitchCursor()
|
|
self.addToGroup(self.pitch)
|
|
self.position = PositionCursor()
|
|
self.addToGroup(self.position)
|
|
api.callbacks.setCursor.append(self.setCursor)
|
|
self.setEnabled(False)
|
|
|
|
def clearItemHighlight(self):
|
|
"""Gets called before a track changes. Most of the time when a new item is inserted/deleted.
|
|
This means the gui track will be recreated and a current highlight on an item might get
|
|
deleted while still on the item. This results in a qt crash
|
|
"""
|
|
Cursor.hightlightEffect = QtWidgets.QGraphicsColorizeEffect() #default strength of the effect is 1.0
|
|
Cursor.hightlightEffect.setColor(QtGui.QColor("cyan"))
|
|
Cursor.hightlightEffect.setStrength(0.7) #opacity of the effect
|
|
|
|
#Cursor.stafflineEffect = QtWidgets.QGraphicsColorizeEffect() #default strength of the effect is 1.0
|
|
#Cursor.stafflineEffect.setColor(QtGui.QColor("cyan"))
|
|
#Cursor.stafflineEffect.setStrength(1) #opacity of the effect
|
|
|
|
Cursor.stafflineEffect = QtWidgets.QGraphicsDropShadowEffect()
|
|
Cursor.stafflineEffect.setColor(QtGui.QColor("black"))
|
|
Cursor.stafflineEffect.setOffset(0,0)
|
|
Cursor.stafflineEffect.setBlurRadius(5)
|
|
|
|
def setCursor(self, cursorExportObject):
|
|
self.cursorExportObject = cursorExportObject
|
|
self.scene().parentView.setFocus()
|
|
|
|
self.pitch.setY(constantsAndConfigs.stafflineGap * cursorExportObject["dotOnLine"] / 2 - 3) #the same as a notehead
|
|
|
|
x = cursorExportObject["tickindex"] / constantsAndConfigs.ticksToPixelRatio
|
|
#Finally shift the Position to the correct track. Trackindex from 0, naturally.
|
|
#self.setPos(x, constantsAndConfigs.timeLineOffsetNoteEditor + cursorExportObject["trackIndex"] * constantsAndConfigs.trackHeight)
|
|
currentGuiTrack = self.scene().tracks[cursorExportObject["trackId"]]
|
|
self.setPos(x, currentGuiTrack.y() )
|
|
|
|
try:
|
|
guiItemAtCursor = currentGuiTrack.itemById(cursorExportObject["itemId"])
|
|
if not guiItemAtCursor.staticItem["completeDuration"] > 0:
|
|
#guiItemAtCursor has no duration. is type guiItemAtCursor.staticItem["type"]
|
|
Cursor.hightlightEffect.setEnabled(True)
|
|
guiItemAtCursor.setGraphicsEffect(Cursor.hightlightEffect)
|
|
else:
|
|
Cursor.hightlightEffect.setEnabled(False)
|
|
|
|
except StopIteration:
|
|
Cursor.hightlightEffect.setEnabled(False) #happens only on startup or appending (which is the same)
|
|
|
|
#Highlight the current staffline
|
|
if currentGuiTrack.staticExportItem["double"] and cursorExportObject["dotOnLine"] in (8, 10, 12, 14, 16):
|
|
lineNumber = int(cursorExportObject["dotOnLine"] / 2) + 1 #gui-stafflines are counted from 0 to n, top to bottom (listindex) while backend stafflines begin on -4
|
|
currentGuiTrack.staffLines[lineNumber].setGraphicsEffect(Cursor.stafflineEffect)
|
|
Cursor.stafflineEffect.setEnabled(True)
|
|
elif cursorExportObject["dotOnLine"] in (-4, -2, 0, 2, 4):
|
|
lineNumber = int(cursorExportObject["dotOnLine"] / 2) + 2 #gui-stafflines are counted from 0 to n, top to bottom (listindex) while backend stafflines begin on -4
|
|
currentGuiTrack.staffLines[lineNumber].setGraphicsEffect(Cursor.stafflineEffect)
|
|
Cursor.stafflineEffect.setEnabled(True)
|
|
else:
|
|
Cursor.stafflineEffect.setEnabled(False)
|
|
|
|
class Playhead(QtWidgets.QGraphicsLineItem):
|
|
def __init__(self, parentScoreScene):
|
|
super().__init__(0, 0, 0, 0) # (x1, y1, x2, y2)
|
|
self.parentScoreScene = parentScoreScene
|
|
p = QtGui.QPen()
|
|
p.setColor(QtGui.QColor("red"))
|
|
p.setCosmetic(True)
|
|
self.setPen(p)
|
|
self.setAcceptHoverEvents(True)
|
|
api.callbacks.setPlaybackTicks.append(self.setCursorPosition)
|
|
api.callbacks.tracksChanged.append(self.setLineToWindowHeigth) #for new tracks
|
|
api.callbacks.updateTempoTrack.append(self.setLineToWindowHeigth)
|
|
self.setFlags(QtWidgets.QGraphicsItem.ItemIsMovable)
|
|
self.setAcceptedMouseButtons(QtCore.Qt.LeftButton)
|
|
self.setZValue(90)
|
|
#self.parentScoreScene.parentView.verticalScrollBar().valueChanged.connect(self.setLineToWindowHeigth)
|
|
#self.hide()
|
|
#self.maxHeight = QtWidgets.QDesktopWidget().geometry().height() #we really hope the screen resolution does not change during the session.
|
|
|
|
def setCursorPosition(self, tickindex:int, playbackStatus:bool):
|
|
"""Set the playhead to the right position, but keep the viewport stable.
|
|
Shift the entire "page" if the cursor becomes invisible because its steps outside the viewport"""
|
|
self.setLineToWindowHeigth()
|
|
x = tickindex / constantsAndConfigs.ticksToPixelRatio
|
|
self.setX(x)
|
|
if constantsAndConfigs.followPlayhead and playbackStatus:
|
|
scenePos = self.parentScoreScene.parentView.mapFromScene(self.pos())
|
|
cursorViewPosX = scenePos.x() #the cursor position in View coordinates
|
|
width = self.parentScoreScene.parentView.geometry().width()
|
|
if cursorViewPosX <= 0 or cursorViewPosX >= width: #"pageflip"
|
|
self.parentScoreScene.parentView.horizontalScrollBar().setValue(x * constantsAndConfigs.zoomFactor)
|
|
|
|
|
|
def _Deprecated__ScrollingVariant___setCursorPosition(self, tickindex):
|
|
"""the tickindex to pixel index on the x axis is a fixed 1:n
|
|
relation. What you see is where you are. No jumps, the playhead
|
|
stays on course."""
|
|
self.setLineToWindowHeigth()
|
|
x = tickindex / constantsAndConfigs.ticksToPixelRatio
|
|
self.setX(x)
|
|
if constantsAndConfigs.followPlayhead and api.playbackStatus():
|
|
#self.parentScoreScene.parentView.centerOn(self) Do not use center on. It centers for Y as well which creates a recursion and the score gets taller and taller.
|
|
#self.scene().parentView.horizontalScrollBar().setValue(x - 150)
|
|
#xV = self.parentScoreScene.parentView.mapFromScene(x,0).x()
|
|
self.parentScoreScene.parentView.horizontalScrollBar().setValue(x - 150) #x does not with zoom levels. the bar drifts away.
|
|
|
|
def setLineToWindowHeigth(self, *args):
|
|
h = self.parentScoreScene.cachedSceneHeight
|
|
self.setLine(0, 0, 0, h) #(x1, y1, x2, y2)
|
|
|
|
def mouseMoveEvent(self, event):
|
|
"""Only allow movement in Y direction.
|
|
Only triggered when dragging."""
|
|
#super().mouseMoveEvent(event) allows free movement through Qt. Don't call that.
|
|
p = event.scenePos().x()
|
|
if p < 0:
|
|
p = 0
|
|
#self.setPos(p, self.scene().parentView.mapToScene(0, 0).y())
|
|
self.setX(p)
|
|
api.seek(p * constantsAndConfigs.ticksToPixelRatio)
|
|
event.accept()
|
|
|
|
def mouseReleaseEvent(self, event):
|
|
if constantsAndConfigs.snapToGrid:
|
|
x = event.scenePos().x() * constantsAndConfigs.ticksToPixelRatio
|
|
p = round(x / constantsAndConfigs.gridRhythm) * constantsAndConfigs.gridRhythm
|
|
if p < 0:
|
|
p = 0
|
|
api.seek(p)
|
|
|
|
def hoverEnterEvent(self, event):
|
|
self.setCursor(QtCore.Qt.SizeHorCursor)
|
|
self.update() #the default implementation calls this. event.accept/ignore has no effect.
|
|
|
|
def hoverLeaveEvent(self, event):
|
|
self.unsetCursor()
|
|
self.update() #the default implementation calls this. event.accept/ignore has no effect.
|
|
|
|
class Selection(QtWidgets.QGraphicsRectItem):
|
|
"""A semi-transparent rectangle that shows the current selection"""
|
|
def __init__(self):
|
|
super().__init__(0, 0, 0, 0)
|
|
|
|
self.validBrush = QtGui.QBrush(QtGui.QColor("grey"))
|
|
self.invalidBrush = QtGui.QBrush(QtGui.QColor("grey"))
|
|
self.invalidBrush.setStyle(QtCore.Qt.DiagCrossPattern)
|
|
|
|
self.setPen(pen)
|
|
self.setOpacity(0.2)
|
|
self.setEnabled(False)
|
|
api.callbacks.setSelection.append(self.setSelection)
|
|
|
|
def setSelection(self, tupleOfCursorExportObjects):
|
|
if tupleOfCursorExportObjects:
|
|
validSelection, topleftCursorObject, bottomRightCursorObject = tupleOfCursorExportObjects
|
|
if validSelection:
|
|
self.setBrush(self.validBrush)
|
|
else:
|
|
self.setBrush(self.invalidBrush)
|
|
|
|
topGuiTrack = self.scene().tracks[topleftCursorObject["trackId"]]
|
|
bottomGuiTrack = self.scene().tracks[bottomRightCursorObject["trackId"]]
|
|
|
|
#y = constantsAndConfigs.timeLineOffsetNoteEditor + topleftCursorObject["trackIndex"] * constantsAndConfigs.trackHeight - constantsAndConfigs.trackHeight/2
|
|
#h = (bottomRightCursorObject["trackIndex"] - topleftCursorObject["trackIndex"]) * constantsAndConfigs.trackHeight + constantsAndConfigs.trackHeight
|
|
|
|
x = topleftCursorObject["tickindex"] / constantsAndConfigs.ticksToPixelRatio
|
|
w = (bottomRightCursorObject["tickindex"] - topleftCursorObject["tickindex"]) / constantsAndConfigs.ticksToPixelRatio
|
|
y = topGuiTrack.y() - constantsAndConfigs.trackHeight/2
|
|
|
|
if bottomGuiTrack.staticExportItem["double"]:
|
|
h = bottomGuiTrack.y() - y + constantsAndConfigs.trackHeight
|
|
else:
|
|
h = bottomGuiTrack.y() - y + constantsAndConfigs.trackHeight/2
|
|
|
|
|
|
self.setRect(0, 0, w-3, h) #substract a few pixels to make it look less ambigious if the last item on the right edge is included or not (it is not)
|
|
self.setPos(x,y)
|
|
self.setVisible(True)
|
|
else: #no selection
|
|
self.setVisible(False)
|
|
|