diff --git a/chordsheet/document.py b/chordsheet/document.py
index 3dd2927..a47a329 100644
--- a/chordsheet/document.py
+++ b/chordsheet/document.py
@@ -18,14 +18,14 @@ class Style:
self.font = kwargs.get('font', 'FreeSans')
self.lineSpacing = kwargs.get('lineSpacing', 1.15)
self.separatorSize = kwargs.get('separatorSize', 5)
+ self.unitWidth = kwargs.get('unitWidth', 10)
self.useIncludedFont = True
self.stringHzSp = 20*self.unit
self.stringHzGap = 2*self.unit
self.stringHeight = 5*self.unit
-
- self.unitWidth = 10*self.unit
+
self.unitHeight = 20*self.unit
self.beatsHeight = 5*self.unit
@@ -58,17 +58,19 @@ class Block:
return NotImplemented
class Document:
- def __init__(self, chordList=None, blockList=None, title=None, composer=None, arranger=None, timeSignature=defaultTimeSignature):
+ def __init__(self, chordList=None, blockList=None, title=None, subtitle=None, composer=None, arranger=None, timeSignature=defaultTimeSignature, tempo=None):
self.chordList = chordList or []
self.blockList = blockList or []
self.title = title or '' # Do not initialise title empty
+ self.subtitle = subtitle
self.composer = composer
self.arranger = arranger
self.timeSignature = timeSignature
+ self.tempo = tempo
def __eq__(self, other):
if isinstance(other, self.__class__):
- textEqual = self.title == other.title and self.composer == other.composer and self.arranger == other.arranger and self.timeSignature == other.timeSignature # check all the text values for equality
+ textEqual = self.title == other.title and self.subtitle == other.subtitle and self.composer == other.composer and self.arranger == other.arranger and self.timeSignature == other.timeSignature and self.tempo == other.tempo # check all the text values for equality
return textEqual and self.chordList == other.chordList and self.blockList == other.blockList
return NotImplemented
@@ -104,9 +106,11 @@ class Document:
self.blockList.append(Block(int(b.find('length').text), chord=blockChord, notes=blockNotes))
self.title = (root.find('title').text if root.find('title') is not None else '') # Do not initialise title empty
+ self.subtitle = (root.find('subtitle').text if root.find('subtitle') is not None else None)
self.composer = (root.find('composer').text if root.find('composer') is not None else None)
self.arranger = (root.find('arranger').text if root.find('arranger') is not None else None)
self.timeSignature = (int(root.find('timesignature').text) if root.find('timesignature') is not None else defaultTimeSignature)
+ self.tempo = (root.find('tempo').text if root.find('tempo') is not None else None)
def newFromXML(filepath):
"""
@@ -124,6 +128,9 @@ class Document:
ET.SubElement(root, "title").text = self.title
+ if self.subtitle is not None:
+ ET.SubElement(root, "subtitle").text = self.subtitle
+
if self.arranger is not None:
ET.SubElement(root, "arranger").text = self.arranger
@@ -131,6 +138,9 @@ class Document:
ET.SubElement(root, "composer").text = self.composer
ET.SubElement(root, "timesignature").text = str(self.timeSignature)
+
+ if self.tempo is not None:
+ ET.SubElement(root, "tempo").text = self.tempo
chordsElement = ET.SubElement(root, "chords")
diff --git a/chordsheet/primitives.py b/chordsheet/primitives.py
index 67bac51..9c6a934 100644
--- a/chordsheet/primitives.py
+++ b/chordsheet/primitives.py
@@ -6,7 +6,7 @@ from reportlab.graphics.shapes import *
def writeText(currentCanvas, style, string, size, vpos, **kwargs):
"""
- Wrapper function to conveniently write text according to my requirements...
+ Wrapper function to conveniently write text and return how much vertical space it took up.
"""
margin = style.leftMargin*style.unit
diff --git a/chordsheet/render.py b/chordsheet/render.py
index aebf210..2cec25c 100644
--- a/chordsheet/render.py
+++ b/chordsheet/render.py
@@ -82,6 +82,7 @@ def guitarChart(currentCanvas, style, chordList, cur_pos):
def chordProgression(currentCanvas, style, document, cur_pos):
margin = style.leftMargin*style.unit
+ unitWidth = style.unitWidth*style.unit
pagesize = style.pageSize
title_height = writeText(currentCanvas, style, "Chord progression", 18, cur_pos, align="left")
@@ -93,11 +94,14 @@ def chordProgression(currentCanvas, style, document, cur_pos):
h_loc = 0
v_loc = 0
- maxWidth = int((((pagesize[0]-(2*margin))/style.unitWidth)//(document.timeSignature*2))*(document.timeSignature*2)) # use integer division to round maxWidth to nearest two bars
-
+ if (unitWidth * document.timeSignature * 2) >= ((pagesize[0]-(2*margin) + 1)): # adding 1 to allow for rounding errors
+ raise Exception("Beat width (unitWidth) is too high. It is {current} pt and can be a maximum of {max} pt".format(current = unitWidth, max = ((pagesize[0]-(2*margin)/(document.timeSignature * 2)))))
+
+ maxWidth = int((((pagesize[0]-(2*margin))/unitWidth)//(document.timeSignature*2))*(document.timeSignature*2)) # use integer division to round maxWidth to nearest two bars
+
for u in range(maxWidth+1):
s = 0
- x = u*style.unitWidth+margin
+ x = u*unitWidth+margin
if u % document.timeSignature == 0:
e = -style.beatsHeight
else:
@@ -105,7 +109,7 @@ def chordProgression(currentCanvas, style, document, cur_pos):
drawVertLine(currentCanvas, s, e, x, h_origin, v_origin)
if u == maxWidth: # Avoid writing beat number after the final line
break
- writeText(currentCanvas, style, str((u % document.timeSignature)+1), style.beatsFontSize, v_origin-style.beatsHeight, hpos=x+style.unitWidth/2)
+ writeText(currentCanvas, style, str((u % document.timeSignature)+1), style.beatsFontSize, v_origin-style.beatsHeight, hpos=x+unitWidth/2)
parsedBlockList = splitBlocks(document.blockList, maxWidth)
@@ -113,12 +117,12 @@ def chordProgression(currentCanvas, style, document, cur_pos):
if h_loc == maxWidth:
v_loc += 1
h_loc = 0
- currentCanvas.rect(h_origin+(h_loc*style.unitWidth), v_origin+(v_loc*style.unitHeight), b.length*style.unitWidth, style.unitHeight)
+ currentCanvas.rect(h_origin+(h_loc*unitWidth), v_origin+(v_loc*style.unitHeight), b.length*unitWidth, style.unitHeight)
if b.notes is not None:
- writeText(currentCanvas, style, b.notes, style.notesFontSize, v_origin+((v_loc+1)*style.unitHeight)-(1.3*style.notesFontSize), hpos=h_origin+((h_loc+b.length/2)*style.unitWidth))
+ writeText(currentCanvas, style, b.notes, style.notesFontSize, v_origin+((v_loc+1)*style.unitHeight)-(1.3*style.notesFontSize), hpos=h_origin+((h_loc+b.length/2)*unitWidth))
v_offset = ((v_loc*style.unitHeight)+style.unitHeight/2)-style.chordNameFontSize/2
if b.chord is not None:
- writeText(currentCanvas, style, b.chord.name, style.chordNameFontSize, v_origin+v_offset, hpos=h_origin+((h_loc+b.length/2)*style.unitWidth))
+ writeText(currentCanvas, style, b.chord.name, style.chordNameFontSize, v_origin+v_offset, hpos=h_origin+((h_loc+b.length/2)*unitWidth))
h_loc += b.length
return v_origin + (v_loc+1)*style.unitHeight + style.beatsHeight + title_height # calculate the height of the generated chart
@@ -139,6 +143,9 @@ def savePDF(document, style, pathToPDF):
if document.title is not None:
curPos += writeText(c, style, document.title, 24, curPos)
+
+ if document.subtitle is not None:
+ curPos += writeText(c, style, document.subtitle, 18, curPos)
if document.composer is not None:
curPos += writeText(c, style, "Composer: {c}".format(c = document.composer), 12, curPos)
@@ -146,6 +153,9 @@ def savePDF(document, style, pathToPDF):
if document.arranger is not None:
curPos += writeText(c, style, "Arranger: {a}".format(a = document.arranger), 12, curPos)
+ if document.tempo is not None:
+ curPos += writeText(c, style, "♩ = {t} bpm".format(t = document.tempo), 12, curPos, align = "left")
+
curPos += style.separatorSize*style.unit
if guitarChartCheck(document.chordList):
diff --git a/gui.py b/gui.py
index 881fa70..b84532b 100755
--- a/gui.py
+++ b/gui.py
@@ -155,6 +155,7 @@ class DocumentWindow(QMainWindow):
self.window.composerLineEdit.setText(self.doc.composer)
self.window.arrangerLineEdit.setText(self.doc.arranger)
self.window.timeSignatureSpinBox.setValue(self.doc.timeSignature)
+ self.window.tempoLineEdit.setText(self.doc.tempo)
self.window.chordTableView.populate(self.doc.chordList)
self.window.blockTableView.populate(self.doc.blockList)
@@ -177,6 +178,8 @@ class DocumentWindow(QMainWindow):
self.window.fontComboBox.setDisabled(True)
self.window.includedFontCheckBox.setChecked(True)
+
+ self.window.beatWidthLineEdit.setText(str(self.style.unitWidth))
def pageSizeAction(self, index):
self.pageSizeSelected = self.window.pageSizeComboBox.itemText(index)
@@ -529,15 +532,25 @@ class DocumentWindow(QMainWindow):
Update the Document object by reading values from the UI.
"""
self.doc.title = self.window.titleLineEdit.text() # Title can be empty string but not None
+ self.doc.subtitle = (self.window.subtitleLineEdit.text() if self.window.subtitleLineEdit.text() else None)
self.doc.composer = (self.window.composerLineEdit.text() if self.window.composerLineEdit.text() else None)
self.doc.arranger = (self.window.arrangerLineEdit.text() if self.window.arrangerLineEdit.text() else None)
-
- self.doc.timeSignature = int(self.window.timeSignatureSpinBox.value())
+ self.doc.tempo = (self.window.tempoLineEdit.text() if self.window.tempoLineEdit.text() else None)
+ self.doc.timeSignature = int(self.window.timeSignatureSpinBox.value()) if self.window.timeSignatureSpinBox.value() else self.doc.timeSignature
+
self.style.pageSize = pageSizeDict[self.pageSizeSelected]
self.style.unit = unitDict[self.unitSelected]
- self.style.leftMargin = int(self.window.leftMarginLineEdit.text())
- self.style.topMargin = int(self.window.topMarginLineEdit.text())
- self.style.lineSpacing = float(self.window.lineSpacingDoubleSpinBox.value())
+ self.style.leftMargin = float(self.window.leftMarginLineEdit.text()) if self.window.leftMarginLineEdit.text() else self.style.leftMargin
+ self.style.topMargin = float(self.window.topMarginLineEdit.text()) if self.window.topMarginLineEdit.text() else self.style.topMargin
+ self.style.lineSpacing = float(self.window.lineSpacingDoubleSpinBox.value()) if self.window.lineSpacingDoubleSpinBox.value() else self.style.lineSpacing
+
+ # make sure the unit width isn't too wide to draw!
+ if self.window.beatWidthLineEdit.text():
+ if (self.style.pageSize[0] - 2 * self.style.leftMargin * mm) >= (float(self.window.beatWidthLineEdit.text()) * 2 * self.doc.timeSignature * mm):
+ self.style.unitWidth = float(self.window.beatWidthLineEdit.text())
+ else:
+ maxBeatWidth = (self.style.pageSize[0] - 2 * self.style.leftMargin * mm) / (2 * self.doc.timeSignature * mm)
+ warning = QMessageBox.warning(self, "Out of range", "Beat width is out of range. It can be a maximum of {}.".format(maxBeatWidth), buttons=QMessageBox.Ok, defaultButton=QMessageBox.Ok)
self.updateChords()
self.updateBlocks()
diff --git a/ui/aboutdialog.ui b/ui/aboutdialog.ui
index 85dd956..f8bc3cc 100644
--- a/ui/aboutdialog.ui
+++ b/ui/aboutdialog.ui
@@ -51,6 +51,9 @@
true
+
+ Qt::AlignLeading|Qt::AlignLeft|Qt::AlignTop
+
-
diff --git a/ui/mainwindow.ui b/ui/mainwindow.ui
index 69370f3..c52fd9d 100644
--- a/ui/mainwindow.ui
+++ b/ui/mainwindow.ui
@@ -6,8 +6,8 @@
0
0
- 761
- 513
+ 1061
+ 646
@@ -42,7 +42,7 @@
430
- 400
+ 600
@@ -104,13 +104,23 @@
-
+
+
+ Subtitle
+
+
+
+ -
+
+
+ -
Composer
- -
+
-
@@ -120,14 +130,14 @@
- -
+
-
Arranger
- -
+
-
@@ -137,18 +147,37 @@
-
-
- -
-
-
-
+
-
+
+
+ Tempo
+
+
+
+ -
+
+
+
+ 0
+ 0
+
+
+
+
+ 60
+ 16777215
+
+
+
+
+ -
- Time signature
+ Time
- -
+
-
@@ -190,6 +219,12 @@
-
+
+
+ 0
+ 0
+
+
Page options
@@ -330,14 +365,44 @@
+ -
+
+
+ Use included FreeSans
+
+
+
- -
-
-
- Use included FreeSans
+
-
+
+
+ Block options
+
+
-
+
+
-
+
+
+ Beat width
+
+
+
+ -
+
+
+
+ 60
+ 16777215
+
+
+
+
+
+
+
-
@@ -739,8 +804,8 @@
0
0
- 298
- 465
+ 598
+ 598
@@ -787,7 +852,7 @@
0
0
- 761
+ 1061
22
@@ -916,14 +981,12 @@
titleLineEdit
composerLineEdit
arrangerLineEdit
- timeSignatureSpinBox
pageSizeComboBox
documentUnitsComboBox
leftMarginLineEdit
topMarginLineEdit
lineSpacingDoubleSpinBox
fontComboBox
- includedFontCheckBox
chordTableView
chordNameLineEdit
guitarVoicingLineEdit