// // File: VLLayout.cpp - Dimensions for lead sheet layout // // Author(s): // // (MN) Matthias Neeracher // // Copyright © 2007-2011 Matthias Neeracher // #include "VLLayout.h" #include "VLSheetViewInternal.h" #include <algorithm> #include <cstdlib> #include <cmath> VLSystemLayout::VLSystemLayout(const VLProperties & prop, float width, int maxMeas) { fDivisions = prop.fDivisions; fNumGroups = prop.fTime.fNum / std::max(prop.fTime.fDenom / 4, 1); fDivPerGroup = fDivisions * prop.fTime.fNum * 4 / (prop.fTime.fDenom * fNumGroups); fClefKeyWidth = kClefX+kClefW+(std::labs(prop.fKey)+1)*kKeyW; fMeasureWidth = fNumGroups*(fDivPerGroup+1)*kNoteW; fNumMeasures = std::min(maxMeas, std::max<int>(1, std::floor((width-fClefKeyWidth) / fMeasureWidth))); } static size_t NextBreak(const VLSong & song, size_t after=0) { size_t propIdx = song.fMeasures[after].fPropIdx; while (++after < song.fMeasures.size()) if (song.fMeasures[after].fBreak || song.fMeasures[after].fPropIdx != propIdx) return after; return song.fMeasures.size(); } VLLayout::VLLayout(const VLSong & song, float width) { size_t nextBreak = song.fMeasures.empty() ? 0 : NextBreak(song); for (size_t meas = 0; meas<song.fMeasures.size(); ) { push_back(VLSystemLayout(song.Properties(meas), width, nextBreak-meas)); meas += back().NumMeasures(); if (meas >= nextBreak) nextBreak = NextBreak(song, nextBreak); } } int VLLayout::FirstMeasure(int system) const { int meas = 0; for (int sys=0; sys<system; ++sys) meas += (*this)[sys].NumMeasures(); return meas; } int VLLayout::SystemForMeasure(int measure) const { int sys; for (sys=0; sys<size(); ++sys) { measure -= (*this)[sys].NumMeasures(); if (measure < 0) break; } return sys; } float VLLayout::MeasurePosition(int measure) const { for (int sys=0; sys<size(); ++sys) { const VLSystemLayout & layout = (*this)[sys]; if (measure < layout.NumMeasures()) return layout.MeasurePosition(measure); measure -= layout.NumMeasures(); } return 0.0f; } float VLLayout::NotePosition(VLLocation at) const { for (int sys=0; sys<size(); ++sys) { const VLSystemLayout & layout = (*this)[sys]; if (at.fMeasure < layout.NumMeasures()) { int div = at.fAt.fNum*4*layout.Divisions() / at.fAt.fDenom; return layout.MeasurePosition(at.fMeasure) + (div + div/layout.DivPerGroup() + 1)*kNoteW; } at.fMeasure -= layout.NumMeasures(); } return 0.0f; } VLFontHandler::~VLFontHandler() { } VLTextLayout::VLTextLayout(VLFontHandler * regular, VLFontHandler * narrow) : fRegularFont(regular), fNarrowFont(narrow) { } void VLTextLayout::AddSyllable(const VLSyllable & syll, float x, bool highlight) { VLLayoutSyll ls; static_cast<VLSyllable &>(ls) = syll; ls.fX = x; ls.fHighlight = highlight; fSyllables.push_back(ls); } typedef std::vector<VLLayoutSyll>::iterator VLSyllIter; #define NARROW_SPACE "\xE2\x80\x89" #define PRE_DASH "-" NARROW_SPACE #define POST_DASH NARROW_SPACE "-" void VLTextLayout::DrawLine(float y) { if (fSyllables.empty()) return; const float kDashW = fRegularFont->Width("-"); const float kSpaceW = fRegularFont->Width(NARROW_SPACE); VLSyllIter syll = fSyllables.begin(); VLSyllIter end = fSyllables.end(); float nextW = fRegularFont->Width(syll->fText.c_str()); float nextX = syll->fX-0.5*nextW; if (syll->fKind & VLSyllable::kHasPrev) fRegularFont->Draw(nextX-fRegularFont->Width(PRE_DASH), y, PRE_DASH, false); while (syll != end) { std::string text = syll->fText; bool hl = syll->fHighlight; VLSyllIter next = syll+1; float curW = nextW; float curX = nextX; while (next != end) { nextW = fRegularFont->Width(next->fText.c_str()); nextX = next->fX-0.5*nextW; if (next->fKind & VLSyllable::kHasPrev) { if (curX+curW+kDashW < nextX) { // // Plenty of space, draw dashes // float dashSpace = 0.5*(nextX-curX-curW-kDashW); fRegularFont->Draw(curX, y, text.c_str(), hl); fRegularFont->Draw(curX+curW+dashSpace, y, "-", hl); goto nextText; } else { // // Fuse & continue // text += next->fText; curW = fRegularFont->Width(text.c_str()); if (++next != end) { nextW = fRegularFont->Width(next->fText.c_str()); nextX = next->fX-0.5*nextW; } } } else { if (curX+curW+kSpaceW < nextX) { // // Enough space, draw regular // fRegularFont->Draw(curX, y, text.c_str(), hl); } else { // // Tight space, draw narrow & adjust // fNarrowFont->Draw(curX, y, text.c_str(), syll->fHighlight); text += NARROW_SPACE; nextX = std::max(nextX, curX+fNarrowFont->Width(text.c_str())); } goto nextText; } } // // At end of line // if ((end-1)->fKind & VLSyllable::kHasNext) text += POST_DASH; fRegularFont->Draw(curX, y, text.c_str(), hl); nextText: syll = next; } }