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.
147 lines
5.9 KiB
147 lines
5.9 KiB
#! /usr/bin/env python3
|
|
# -*- coding: utf-8 -*-
|
|
"""
|
|
Copyright 2022, Nils Hilbricht, Germany ( https://www.hilbricht.net )
|
|
|
|
This file is part of the Laborejo Software Suite ( https://www.laborejo.org ),
|
|
|
|
Laborejo2 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/>.
|
|
"""
|
|
|
|
import logging; logger = logging.getLogger(__name__); logger.info("import")
|
|
|
|
#Standard Library Modules
|
|
|
|
#Third Party Modules
|
|
|
|
#Template Modules
|
|
import template.engine.pitch as pitch
|
|
import template.engine.duration as duration
|
|
|
|
#Our modules
|
|
class Cursor:
|
|
"""There is only one cursor in the program.
|
|
Each track has its own state with its own gap, tickindex etc.
|
|
but the cursor is unique.
|
|
Used for pitch and states which apply to all tracks, like
|
|
"insert next note with dot?"
|
|
|
|
The cursor moves 'on stafflines' not through the keysig. Adaption to the keysig is done
|
|
by the api like in insertCursorCode or insertKeysignature
|
|
"""
|
|
def __init__(self, score):
|
|
self.score = score
|
|
self.pitchindex = pitch.ly2pitch["c'"]
|
|
self.prevailingBaseDuration = duration.D4
|
|
self.persistentPrevailingDot = False #For the sake of simplicity this is only a bool. No "two prevailing dots"
|
|
self.oneTimePrevailingDotInverter = False #effects persistentPrevailingDot
|
|
self.cursorWasMovedAfterChoosingPitchViaMidi = False #this flag enables the api to check if the cursor was set with cursor commands (=arrow keys) after it was set by midi. If not the api can insert the literal midi pitch
|
|
|
|
def up(self):
|
|
"""Move the cursor/pitchindex up, +50.
|
|
The cursor always goes up and down the c major scale.
|
|
Call with Cursor.up(). Class method """
|
|
self.pitchindex += pitch.STEP
|
|
self.correctPitchindex()
|
|
|
|
def down(self):
|
|
"""Move the cursor/pitchindex down, -50.
|
|
The cursor always goes up and down the c major scale.
|
|
Call with Cursor.down(). Class method"""
|
|
self.pitchindex -= pitch.STEP
|
|
self.correctPitchindex()
|
|
|
|
def upOctave(self):
|
|
"""Move the cursor/pitchindex up, +350.
|
|
Call with Cursor.upOctave(). Class method"""
|
|
self.pitchindex += pitch.OCTAVE
|
|
self.correctPitchindex()
|
|
|
|
def downOctave(self):
|
|
"""Move the cursor/pitchindex down, -350.
|
|
Call with Cursor.downOctave(). Class method"""
|
|
self.pitchindex -= pitch.OCTAVE
|
|
self.correctPitchindex()
|
|
|
|
def correctPitchindex(self):
|
|
self.cursorWasMovedAfterChoosingPitchViaMidi = True
|
|
if self.pitchindex >= pitch.MAX:
|
|
self.pitchindex = pitch.MAX
|
|
elif self.pitchindex <= pitch.MIN:
|
|
self.pitchindex = pitch.MIN
|
|
else:
|
|
self.pitchindex = pitch.toWhite[self.pitchindex]
|
|
|
|
def getPrevailingDot(self):
|
|
"""0 or 1.
|
|
Also toggles oneTimePrevailingDotInverter"""
|
|
if self.persistentPrevailingDot:
|
|
if self.oneTimePrevailingDotInverter:
|
|
self.oneTimePrevailingDotInverter = not self.oneTimePrevailingDotInverter
|
|
return 0
|
|
else:
|
|
return 1
|
|
else:
|
|
if self.oneTimePrevailingDotInverter:
|
|
self.oneTimePrevailingDotInverter = not self.oneTimePrevailingDotInverter
|
|
return 1
|
|
else:
|
|
return 0
|
|
|
|
def distanceInDiatonicStepsFromTrebleB(self):
|
|
return pitch.distanceInDiatonicSteps(pitch.ly2pitch["b'"], self.pitchindex) #offset from the middle line in treble clef h', which is 0. c'' is -1, a' is +1
|
|
|
|
def asDotOnLine(self, clef):
|
|
return self.distanceInDiatonicStepsFromTrebleB() + clef.asDotOnLineOffset
|
|
|
|
def exportObject(self, trackState):
|
|
"""Every time the cursor moves this will be called.
|
|
|
|
return a dict which is a good layout
|
|
export basis. For example for GUI frontends. They don't have to
|
|
parse and calculate their own values in slow pure Python then.
|
|
|
|
This is for GUIs etc. not for saving to JSOn. The cursor is not
|
|
saved.
|
|
"""
|
|
tempoItem = trackState.track.parentData.tempoTrack.tempoAtTickPosition(trackState.tickindex) #TODO: maybe get that from a cached version.
|
|
item = trackState.track.currentItem()
|
|
block = trackState.track.currentBlock()
|
|
|
|
|
|
return {
|
|
"type": "Cursor",
|
|
"trackIndex": trackState.index(),
|
|
"track" : trackState.track,
|
|
"trackId" : id(trackState.track),
|
|
"trackName" : trackState.track.name,
|
|
"cboxMidiOutUuid" : trackState.track.sequencerInterface.cboxMidiOutUuid, #used for midi throught. Step midi shall produce sound through the current track.
|
|
"midiChannel" : trackState.midiChannel(), #zero based
|
|
"trackId" : id(trackState.track),
|
|
"position" : trackState.position(),
|
|
"tickindex" : trackState.tickindex,
|
|
"dotOnLine": self.asDotOnLine(trackState.clef()),
|
|
"pitch" : self.pitchindex,
|
|
"lilypondPitch" : pitch.pitch2ly[self.pitchindex],
|
|
"appending" : trackState.isAppending(),
|
|
"blockindex" : trackState.blockindex,
|
|
"block" : block,
|
|
"blockName" : block.name,
|
|
"localCursorIndex" : block.localCursorIndex,
|
|
"itemId" : id(item),
|
|
"item" : item,
|
|
"tempoUnitsPerMinute" : tempoItem.unitsPerMinute,
|
|
"tempoReferenceTicks" : tempoItem.referenceTicks,
|
|
"tempoGraphType" : tempoItem.graphType,
|
|
}
|
|
|