1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828 |
- /*
- * draw_pat.cpp
- * ------------
- * Purpose: Code for drawing the pattern data.
- * Notes : Also used for updating the status bar.
- * Authors: Olivier Lapicque
- * OpenMPT Devs
- * The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
- */
- #include "stdafx.h"
- #include "Mptrack.h"
- #include "Mainfrm.h"
- #include "Moddoc.h"
- #include "dlg_misc.h"
- #include "Globals.h"
- #include "View_pat.h"
- #include "EffectVis.h"
- #include "ChannelManagerDlg.h"
- #include "../soundlib/tuning.h"
- #include "../soundlib/mod_specifications.h"
- #include "../soundlib/Tables.h"
- #include "../soundlib/plugins/PlugInterface.h"
- #include "../common/mptStringBuffer.h"
- #include "EffectInfo.h"
- #include "PatternFont.h"
- OPENMPT_NAMESPACE_BEGIN
- // Headers
- enum
- {
- ROWHDR_WIDTH = 32, // Row header
- COLHDR_HEIGHT = 16 + 4, // Column header (name + color)
- VUMETERS_HEIGHT = 13, // Height of vu-meters
- PLUGNAME_HEIGHT = 16, // Height of plugin names
- VUMETERS_BMPWIDTH = 32,
- VUMETERS_BMPHEIGHT = 10,
- VUMETERS_MEDWIDTH = 24,
- VUMETERS_LOWIDTH = 16,
- };
- enum
- {
- COLUMN_BITS_NONE = 0x00,
- COLUMN_BITS_NOTE = 0x01,
- COLUMN_BITS_INSTRUMENT = 0x02,
- COLUMN_BITS_VOLUME = 0x04,
- COLUMN_BITS_FXCMD = 0x08,
- COLUMN_BITS_FXPARAM = 0x10,
- COLUMN_BITS_FXCMDANDPARAM = 0x18,
- COLUMN_BITS_ALLCOLUMNS = 0x1F,
- COLUMN_BITS_UNKNOWN = 0x20, // Appears to be unused
- COLUMN_BITS_ALL = 0x3F,
- COLUMN_BITS_SKIP = 0x40,
- COLUMN_BITS_INVISIBLE = 0x80,
- };
- /////////////////////////////////////////////////////////////////////////////
- // Effect colour codes
- // EffectType => ModColor mapping
- static constexpr int effectColors[] =
- {
- 0,
- MODCOLOR_GLOBALS,
- MODCOLOR_VOLUME,
- MODCOLOR_PANNING,
- MODCOLOR_PITCH,
- };
- static_assert(std::size(effectColors) == MAX_EFFECT_TYPE);
- /////////////////////////////////////////////////////////////////////////////
- // CViewPattern Drawing Implementation
- static uint8 HighlightColor(int c0, int c1)
- {
- int cf0 = 0xC0 - (c1 >> 2) - (c0 >> 3);
- Limit(cf0, 0x40, 0xC0);
- int cf1 = 0x100 - cf0;
- return static_cast<uint8>((c0 * cf0 + c1 * cf1) >> 8);
- }
- static void MixColors(CFastBitmap &dib, ModColor target, ModColor src1, ModColor src2)
- {
- const auto c1 = TrackerSettings::Instance().rgbCustomColors[src1], c2 = TrackerSettings::Instance().rgbCustomColors[src2];
- auto r = HighlightColor(GetRValue(c1), GetRValue(c2));
- auto g = HighlightColor(GetGValue(c1), GetGValue(c2));
- auto b = HighlightColor(GetBValue(c1), GetBValue(c2));
- dib.SetColor(target, RGB(r, g, b));
- }
- void CViewPattern::UpdateColors()
- {
- m_Dib.SetAllColors(0, MAX_MODCOLORS, TrackerSettings::Instance().rgbCustomColors.data());
- MixColors(m_Dib, MODCOLOR_2NDHIGHLIGHT, MODCOLOR_BACKHILIGHT, MODCOLOR_BACKNORMAL);
- MixColors(m_Dib, MODCOLOR_DEFAULTVOLUME, MODCOLOR_VOLUME, MODCOLOR_BACKNORMAL);
- MixColors(m_Dib, MODCOLOR_DUMMYCOMMAND, MODCOLOR_TEXTNORMAL, MODCOLOR_BACKNORMAL);
- m_Dib.SetBlendColor(TrackerSettings::Instance().rgbCustomColors[MODCOLOR_BLENDCOLOR]);
- }
- bool CViewPattern::UpdateSizes()
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- int oldx = m_szCell.cx, oldy = m_szCell.cy;
- m_szHeader.cx = ROWHDR_WIDTH;
- m_szHeader.cy = COLHDR_HEIGHT;
- m_szPluginHeader.cx = 0;
- m_szPluginHeader.cy = m_Status[psShowPluginNames] ? MulDiv(PLUGNAME_HEIGHT, m_nDPIy, 96) : 0;
- if(m_Status[psShowVUMeters]) m_szHeader.cy += VUMETERS_HEIGHT;
- m_szCell.cx = 4 + pfnt->nEltWidths[0];
- if (m_nDetailLevel >= PatternCursor::instrColumn) m_szCell.cx += pfnt->nEltWidths[1];
- if (m_nDetailLevel >= PatternCursor::volumeColumn) m_szCell.cx += pfnt->nEltWidths[2];
- if (m_nDetailLevel >= PatternCursor::effectColumn) m_szCell.cx += pfnt->nEltWidths[3] + pfnt->nEltWidths[4];
- m_szCell.cy = pfnt->nHeight;
- m_szHeader.cx = MulDiv(m_szHeader.cx, m_nDPIx, 96);
- m_szHeader.cy = MulDiv(m_szHeader.cy, m_nDPIy, 96);
- m_szHeader.cy += m_szPluginHeader.cy;
- if(oldy != m_szCell.cy)
- {
- m_Dib.SetSize(m_Dib.GetWidth(), m_szCell.cy);
- }
- return (oldx != m_szCell.cx || oldy != m_szCell.cy);
- }
- UINT CViewPattern::GetColumnOffset(PatternCursor::Columns column) const
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- LimitMax(column, PatternCursor::lastColumn);
- UINT offset = 0;
- for(int i = PatternCursor::firstColumn; i < column; i++)
- offset += pfnt->nEltWidths[i];
- return offset;
- }
- int CViewPattern::GetSmoothScrollOffset() const
- {
- if((TrackerSettings::Instance().m_dwPatternSetup & PATTERN_SMOOTHSCROLL) != 0 // Actually using the smooth scroll feature
- && (m_Status & (psFollowSong | psDragActive)) == psFollowSong // Not drawing a selection during playback
- && (m_nMidRow != 0 || GetYScrollPos() > 0) // If active row is not centered, only scroll when display position is actually not at the top
- && IsLiveRecord() // Actually playing live (not paused or stepping)
- && m_nNextPlayRow != m_nPlayRow) // Don't scroll if we stay on the same row
- {
- uint32 tick = m_nPlayTick;
- // Avoid jerky animation with backwards-going patterns
- if(m_nNextPlayRow == m_nPlayRow - 1) tick = m_nTicksOnRow - m_nPlayTick - 1;
- return Util::muldivr_unsigned(m_szCell.cy, tick, std::max(1u, m_nTicksOnRow));
- }
- return 0;
- }
- void CViewPattern::UpdateView(UpdateHint hint, CObject *pObj)
- {
- if(pObj == this)
- {
- return;
- }
- if(hint.GetType()[HINT_MPTOPTIONS])
- {
- PatternFont::UpdateFont(m_hWnd);
- UpdateColors();
- UpdateSizes();
- UpdateScrollSize();
- InvalidatePattern(true, true);
- return;
- }
- const auto generalHint = hint.ToType<GeneralHint>();
- if(generalHint.GetType()[HINT_MODTYPE | HINT_MODCHANNELS])
- {
- InvalidateChannelsHeaders();
- UpdateScrollSize();
- }
- if(generalHint.GetType()[HINT_MODTYPE])
- {
- // If sequence and pattern view became inconsistent (e.g. due to rearranging patterns during cleanup), synchronize to order list again
- const auto &order = Order();
- const ORDERINDEX ord = GetCurrentOrder();
- if(order.IsValidPat(ord) && order.at(ord) != m_nPattern)
- SetCurrentPattern(order.at(ord));
- }
- if(generalHint.GetType()[HINT_MODCHANNELS]
- && m_quickChannelProperties.m_hWnd
- && pObj != &m_quickChannelProperties
- && (generalHint.GetChannel() >= GetDocument()->GetNumChannels() || generalHint.GetChannel() == m_quickChannelProperties.GetChannel()))
- {
- m_quickChannelProperties.UpdateDisplay();
- }
- const PatternHint patternHint = hint.ToType<PatternHint>();
- const PATTERNINDEX updatePat = patternHint.GetPattern();
- if(hint.GetType() == HINT_PATTERNDATA
- && m_nPattern != updatePat
- && updatePat != 0
- && updatePat != GetNextPattern()
- && updatePat != GetPrevPattern())
- return;
- if(patternHint.GetType()[HINT_MODTYPE | HINT_PATTERNDATA])
- {
- InvalidatePattern(false, true);
- } else if(patternHint.GetType()[HINT_PATTERNROW])
- {
- InvalidateRow(static_cast<const RowHint &>(hint).GetRow());
- }
- }
- POINT CViewPattern::GetPointFromPosition(PatternCursor cursor) const
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- POINT pt;
- int xofs = GetXScrollPos();
- int yofs = GetYScrollPos();
- PatternCursor::Columns imax = cursor.GetColumnType();
- LimitMax(imax, PatternCursor::lastColumn);
- // if(imax > m_nDetailLevel)
- // {
- // // Extend to next channel
- // imax = PatternCursor::firstColumn;
- // cursor.Move(0, 1, 0);
- // }
- pt.x = (cursor.GetChannel() - xofs) * GetChannelWidth();
- for(int i = 0; i < imax; i++)
- {
- pt.x += pfnt->nEltWidths[i];
- }
- if (pt.x < 0) pt.x = 0;
- pt.x += Util::ScalePixels(ROWHDR_WIDTH, m_hWnd);
- pt.y = (cursor.GetRow() - yofs + m_nMidRow) * m_szCell.cy;
- if (pt.y < 0) pt.y = 0;
- pt.y += m_szHeader.cy;
- return pt;
- }
- PatternCursor CViewPattern::GetPositionFromPoint(POINT pt) const
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- int xofs = GetXScrollPos();
- int yofs = GetYScrollPos();
- int x = xofs + (pt.x - m_szHeader.cx) / GetChannelWidth();
- if (pt.x < m_szHeader.cx) x = (xofs) ? xofs - 1 : 0;
- int y = yofs - m_nMidRow + (pt.y - m_szHeader.cy + GetSmoothScrollOffset()) / m_szCell.cy;
- if (y < 0) y = 0;
- int xx = (pt.x - m_szHeader.cx) % GetChannelWidth(), dx = 0;
- int imax = 4;
- if (imax > (int)m_nDetailLevel + 1) imax = m_nDetailLevel + 1;
- int i = 0;
- for (i=0; i<imax; i++)
- {
- dx += pfnt->nEltWidths[i];
- if(xx < dx)
- break;
- }
- return PatternCursor(static_cast<ROWINDEX>(y), static_cast<CHANNELINDEX>(x), static_cast<PatternCursor::Columns>(i));
- }
- void CViewPattern::DrawLetter(int x, int y, char letter, int sizex, int ofsx)
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- if(pfnt->dibASCII)
- {
- if(32 <= letter && letter <= 127)
- {
- m_Dib.TextBlt(x, y, sizex, pfnt->spacingY, (((unsigned char)letter) * pfnt->nNoteWidth[0]) + ofsx, 0, pfnt->dibASCII);
- return;
- }
- }
- int srcx = pfnt->nSpaceX, srcy = pfnt->nSpaceY;
- if ((letter >= '0') && (letter <= '9'))
- {
- srcx = pfnt->nNumX;
- srcy = pfnt->nNumY + (letter - '0') * pfnt->spacingY;
- } else
- if ((letter >= 'A') && (letter < 'N'))
- {
- srcx = pfnt->nAlphaAM_X;
- srcy = pfnt->nAlphaAM_Y + (letter - 'A') * pfnt->spacingY;
- } else
- if ((letter >= 'N') && (letter <= 'Z'))
- {
- srcx = pfnt->nAlphaNZ_X;
- srcy = pfnt->nAlphaNZ_Y + (letter - 'N') * pfnt->spacingY;
- } else
- switch(letter)
- {
- case '?':
- srcx = pfnt->nAlphaNZ_X;
- srcy = pfnt->nAlphaNZ_Y + 13 * pfnt->spacingY;
- break;
- case '#':
- srcx = pfnt->nAlphaAM_X;
- srcy = pfnt->nAlphaAM_Y + 13 * pfnt->spacingY;
- break;
- case '\\':
- srcx = pfnt->nAlphaNZ_X;
- srcy = pfnt->nAlphaNZ_Y + 14 * pfnt->spacingY;
- break;
- case ':':
- srcx = pfnt->nAlphaNZ_X;
- srcy = pfnt->nAlphaNZ_Y + 15 * pfnt->spacingY;
- break;
- case '*':
- srcx = pfnt->nAlphaNZ_X;
- srcy = pfnt->nAlphaNZ_Y + 16 * pfnt->spacingY;
- break;
- case ' ':
- srcx = pfnt->nSpaceX;
- srcy = pfnt->nSpaceY;
- break;
- case 'b':
- srcx = pfnt->nAlphaAM_X;
- srcy = pfnt->nAlphaAM_Y + 14 * pfnt->spacingY;
- break;
- case '-':
- srcx = pfnt->nAlphaAM_X;
- srcy = pfnt->nAlphaAM_Y + 15 * pfnt->spacingY;
- break;
- case '+':
- srcx = pfnt->nAlphaAM_X;
- srcy = pfnt->nAlphaAM_Y + 16 * pfnt->spacingY;
- break;
- case 'd':
- srcx = pfnt->nAlphaAM_X;
- srcy = pfnt->nAlphaAM_Y + 17 * pfnt->spacingY;
- break;
- case '.':
- srcx = pfnt->nNoteX;
- srcy = pfnt->nNoteY;
- break;
- }
- m_Dib.TextBlt(x, y, sizex, pfnt->spacingY, srcx+ofsx, srcy, pfnt->dib);
- }
- void CViewPattern::DrawLetter(int x, int y, wchar_t letter, int sizex, int ofsx)
- {
- DrawLetter(x, y, mpt::unsafe_char_convert<char>(letter), sizex, ofsx);
- }
- #if MPT_CXX_AT_LEAST(20)
- void CViewPattern::DrawLetter(int x, int y, char8_t letter, int sizex, int ofsx)
- {
- DrawLetter(x, y, mpt::unsafe_char_convert<char>(letter), sizex, ofsx);
- }
- #endif
- static MPT_FORCEINLINE void DrawPadding(CFastBitmap &dib, const PATTERNFONT *pfnt, int x, int y, int col)
- {
- if(pfnt->padding[col])
- dib.TextBlt(x + pfnt->nEltWidths[col] - pfnt->padding[col], y, pfnt->padding[col], pfnt->spacingY, pfnt->nClrX + pfnt->nEltWidths[col] - pfnt->padding[col], pfnt->nClrY, pfnt->dib);
- }
- void CViewPattern::DrawNote(int x, int y, UINT note, CTuning* pTuning)
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- UINT xsrc = pfnt->nNoteX, ysrc = pfnt->nNoteY, dx = pfnt->nEltWidths[0];
- if (!note)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, xsrc, ysrc, pfnt->dib);
- } else
- if (note == NOTE_NOTECUT)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, xsrc, ysrc + 13*pfnt->spacingY, pfnt->dib);
- } else
- if (note == NOTE_KEYOFF)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, xsrc, ysrc + 14*pfnt->spacingY, pfnt->dib);
- } else
- if(note == NOTE_FADE)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, xsrc, ysrc + 17*pfnt->spacingY, pfnt->dib);
- } else
- if(note == NOTE_PC)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, xsrc, ysrc + 15*pfnt->spacingY, pfnt->dib);
- } else
- if(note == NOTE_PCS)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, xsrc, ysrc + 16*pfnt->spacingY, pfnt->dib);
- } else
- {
- if(pTuning)
- {
- // Drawing custom note names
- std::wstring noteStr = mpt::ToWide(pTuning->GetNoteName(static_cast<Tuning::NOTEINDEXTYPE>(note - NOTE_MIDDLEC)));
- if(noteStr.size() < 3)
- noteStr.resize(3, L' ');
-
- DrawLetter(x, y, noteStr[0], pfnt->nNoteWidth[0], 0);
- DrawLetter(x + pfnt->nNoteWidth[0], y, noteStr[1], pfnt->nNoteWidth[1], 0);
- DrawLetter(x + pfnt->nNoteWidth[0] + pfnt->nNoteWidth[1], y, noteStr[2], pfnt->nOctaveWidth, 0);
- } else
- {
- // Original
- UINT o = (note - NOTE_MIN) / 12; //Octave
- UINT n = (note - NOTE_MIN) % 12; //Note
- // Hack for default pattern font, allowing for sharps
- if(TrackerSettings::Instance().accidentalFlats)
- {
- DrawLetter(x, y, NoteNamesFlat[n][0], pfnt->nNoteWidth[0], 0);
- DrawLetter(x + pfnt->nNoteWidth[0], y, NoteNamesFlat[n][1], pfnt->nNoteWidth[1], 0);
- } else
- {
- m_Dib.TextBlt(x, y, pfnt->nNoteWidth[0] + pfnt->nNoteWidth[1], pfnt->spacingY, xsrc, ysrc+(n+1)*pfnt->spacingY, pfnt->dib);
- }
- if(o <= 15)
- m_Dib.TextBlt(x + pfnt->nNoteWidth[0] + pfnt->nNoteWidth[1], y, pfnt->nOctaveWidth, pfnt->spacingY,
- pfnt->nNumX, pfnt->nNumY+o*pfnt->spacingY, pfnt->dib);
- else
- DrawLetter(x + pfnt->nNoteWidth[0] + pfnt->nNoteWidth[1], y, '?', pfnt->nOctaveWidth);
- }
- }
- DrawPadding(m_Dib, pfnt, x, y, 0);
- }
- void CViewPattern::DrawInstrument(int x, int y, UINT instr)
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- if (instr)
- {
- UINT dx = pfnt->nInstrHiWidth;
- if (instr < 100)
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, pfnt->nNumX+pfnt->nInstrOfs, pfnt->nNumY+(instr / 10)*pfnt->spacingY, pfnt->dib);
- } else
- {
- m_Dib.TextBlt(x, y, dx, pfnt->spacingY, pfnt->nNum10X+pfnt->nInstr10Ofs, pfnt->nNum10Y+((instr-100) / 10)*pfnt->spacingY, pfnt->dib);
- }
- m_Dib.TextBlt(x+dx, y, pfnt->nEltWidths[1]-dx, pfnt->spacingY, pfnt->nNumX+pfnt->paramLoMargin, pfnt->nNumY+(instr % 10)*pfnt->spacingY, pfnt->dib);
- } else
- {
- m_Dib.TextBlt(x, y, pfnt->nEltWidths[1], pfnt->spacingY, pfnt->nClrX+pfnt->nEltWidths[0], pfnt->nClrY, pfnt->dib);
- }
- DrawPadding(m_Dib, pfnt, x, y, 1);
- }
- void CViewPattern::DrawVolumeCommand(int x, int y, const ModCommand &mc, bool drawDefaultVolume)
- {
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- if(mc.IsPcNote())
- {
- //If note is parameter control note, drawing volume command differently.
- const int val = std::min(ModCommand::maxColumnValue, static_cast<int>(mc.GetValueVolCol()));
- if(pfnt->pcParamMargin) m_Dib.TextBlt(x, y, pfnt->pcParamMargin, pfnt->spacingY, pfnt->nClrX, pfnt->nClrY, pfnt->dib);
- m_Dib.TextBlt(x + pfnt->pcParamMargin, y, pfnt->nVolCmdWidth, pfnt->spacingY,
- pfnt->nNumX, pfnt->nNumY+(val / 100)*pfnt->spacingY, pfnt->dib);
- m_Dib.TextBlt(x+pfnt->nVolCmdWidth, y, pfnt->nVolHiWidth, pfnt->spacingY,
- pfnt->nNumX, pfnt->nNumY+((val / 10)%10)*pfnt->spacingY, pfnt->dib);
- m_Dib.TextBlt(x+pfnt->nVolCmdWidth+pfnt->nVolHiWidth, y, pfnt->nEltWidths[2]-(pfnt->nVolCmdWidth+pfnt->nVolHiWidth), pfnt->spacingY,
- pfnt->nNumX, pfnt->nNumY+(val % 10)*pfnt->spacingY, pfnt->dib);
- } else
- {
- ModCommand::VOLCMD volcmd = mc.volcmd;
- int vol = (mc.vol & 0x7F);
- if(drawDefaultVolume)
- {
- // Displaying sample default volume if there is no volume command.
- volcmd = VOLCMD_VOLUME;
- vol = GetDefaultVolume(mc);
- }
- if(volcmd != VOLCMD_NONE && volcmd < MAX_VOLCMDS)
- {
- m_Dib.TextBlt(x, y, pfnt->nVolCmdWidth, pfnt->spacingY,
- pfnt->nVolX, pfnt->nVolY + volcmd * pfnt->spacingY, pfnt->dib);
- m_Dib.TextBlt(x+pfnt->nVolCmdWidth, y, pfnt->nVolHiWidth, pfnt->spacingY,
- pfnt->nNumX, pfnt->nNumY + (vol / 10) * pfnt->spacingY, pfnt->dib);
- m_Dib.TextBlt(x+pfnt->nVolCmdWidth + pfnt->nVolHiWidth, y, pfnt->nEltWidths[2] - (pfnt->nVolCmdWidth + pfnt->nVolHiWidth), pfnt->spacingY,
- pfnt->nNumX, pfnt->nNumY + (vol % 10) * pfnt->spacingY, pfnt->dib);
- } else
- {
- int srcx = pfnt->nEltWidths[0] + pfnt->nEltWidths[1];
- m_Dib.TextBlt(x, y, pfnt->nEltWidths[2], pfnt->spacingY, pfnt->nClrX+srcx, pfnt->nClrY, pfnt->dib);
- }
- }
- DrawPadding(m_Dib, pfnt, x, y, 2);
- }
- void CViewPattern::OnDraw(CDC *pDC)
- {
- CMainFrame *pMainFrm = CMainFrame::GetMainFrame();
- CHAR s[256];
- CRect rcClient, rect, rc;
- const CModDoc *pModDoc;
- MPT_ASSERT(pDC);
- UpdateSizes();
- if ((pModDoc = GetDocument()) == nullptr) return;
-
- const int vuHeight = MulDiv(VUMETERS_HEIGHT, m_nDPIy, 96);
- const int colHeight = MulDiv(COLHDR_HEIGHT, m_nDPIy, 96);
- const int chanColorHeight = MulDiv(4, m_nDPIy, 96);
- const int chanColorOffset = MulDiv(2, m_nDPIy, 96);
- const int recordInsX = MulDiv(3, m_nDPIx, 96);
- const bool doSmoothScroll = (TrackerSettings::Instance().m_dwPatternSetup & PATTERN_SMOOTHSCROLL) != 0;
- GetClientRect(&rcClient);
- HDC hdc;
- HBITMAP oldBitmap = NULL;
- if(doSmoothScroll)
- {
- if(rcClient != m_oldClient)
- {
- m_offScreenBitmap.DeleteObject();
- m_offScreenDC.DeleteDC();
- m_offScreenDC.CreateCompatibleDC(pDC);
- m_offScreenBitmap.CreateCompatibleBitmap(pDC, rcClient.Width(), rcClient.Height());
- m_oldClient = rcClient;
- }
- hdc = m_offScreenDC;
- oldBitmap = SelectBitmap(hdc, m_offScreenBitmap);
- } else
- {
- hdc = pDC->m_hDC;
- }
- const auto dcBrush = GetStockBrush(DC_BRUSH);
- const auto faceColor = GetSysColor(COLOR_BTNFACE);
- const auto shadowColor = GetSysColor(COLOR_BTNSHADOW);
- const auto textColor = GetSysColor(COLOR_BTNTEXT);
- CHANNELINDEX xofs = static_cast<CHANNELINDEX>(GetXScrollPos());
- ROWINDEX yofs = static_cast<ROWINDEX>(GetYScrollPos());
- const CSoundFile &sndFile = pModDoc->GetSoundFile();
- UINT nColumnWidth = m_szCell.cx;
- UINT ncols = sndFile.GetNumChannels();
- int xpaint = m_szHeader.cx;
- int ypaint = rcClient.top + m_szHeader.cy - GetSmoothScrollOffset();
- const auto &order = Order();
- const ORDERINDEX ordCount = Order().GetLength();
- if (m_nMidRow)
- {
- if (yofs >= m_nMidRow)
- {
- yofs -= m_nMidRow;
- } else
- {
- UINT nSkip = m_nMidRow - yofs;
- PATTERNINDEX nPrevPat = PATTERNINDEX_INVALID;
- // Display previous pattern
- if (TrackerSettings::Instance().m_dwPatternSetup & PATTERN_SHOWPREVIOUS)
- {
- if(m_nOrder > 0 && m_nOrder < ordCount)
- {
- ORDERINDEX prevOrder = order.GetPreviousOrderIgnoringSkips(m_nOrder);
- //Skip +++ items
- if(m_nOrder < order.size() && order[m_nOrder] == m_nPattern)
- {
- nPrevPat = order[prevOrder];
- }
- }
- }
- if(sndFile.Patterns.IsValidPat(nPrevPat))
- {
- ROWINDEX nPrevRows = sndFile.Patterns[nPrevPat].GetNumRows();
- ROWINDEX n = std::min(static_cast<ROWINDEX>(nSkip), nPrevRows);
- ypaint += (nSkip - n) * m_szCell.cy;
- rect.SetRect(0, m_szHeader.cy, nColumnWidth * ncols + m_szHeader.cx, ypaint - 1);
- m_Dib.SetBlendMode(true);
- DrawPatternData(hdc, nPrevPat, false, false,
- nPrevRows - n, nPrevRows, xofs, rcClient, &ypaint);
- m_Dib.SetBlendMode(false);
- } else
- {
- ypaint += nSkip * m_szCell.cy;
- rect.SetRect(0, m_szHeader.cy, nColumnWidth * ncols + m_szHeader.cx, ypaint - 1);
- }
- if ((rect.bottom > rect.top) && (rect.right > rect.left))
- {
- ::SetDCBrushColor(hdc, faceColor);
- ::FillRect(hdc, &rect, dcBrush);
- auto shadowRect = rect;
- shadowRect.top = shadowRect.bottom++;
- ::SetDCBrushColor(hdc, shadowColor);
- ::FillRect(hdc, &shadowRect, dcBrush);
- }
- yofs = 0;
- }
- }
- UINT nrows = sndFile.Patterns.IsValidPat(m_nPattern) ? sndFile.Patterns[m_nPattern].GetNumRows() : 0;
- int ypatternend = ypaint + (nrows-yofs)*m_szCell.cy;
- DrawPatternData(hdc, m_nPattern, true, (pMainFrm->GetModPlaying() == pModDoc),
- yofs, nrows, xofs, rcClient, &ypaint);
- // Display next pattern
- if ((TrackerSettings::Instance().m_dwPatternSetup & PATTERN_SHOWPREVIOUS) && (ypaint < rcClient.bottom) && (ypaint == ypatternend))
- {
- int nVisRows = (rcClient.bottom - ypaint + m_szCell.cy - 1) / m_szCell.cy;
- if ((nVisRows > 0) && (m_nMidRow))
- {
- PATTERNINDEX nNextPat = PATTERNINDEX_INVALID;
- ORDERINDEX nNextOrder = order.GetNextOrderIgnoringSkips(m_nOrder);
- if(nNextOrder == m_nOrder) nNextOrder = ORDERINDEX_INVALID;
- //Ignore skip items(+++) from sequence.
- if(m_nOrder < ordCount && nNextOrder < ordCount && order[m_nOrder] == m_nPattern)
- {
- nNextPat = order[nNextOrder];
- }
- if(sndFile.Patterns.IsValidPat(nNextPat))
- {
- ROWINDEX nNextRows = sndFile.Patterns[nNextPat].GetNumRows();
- ROWINDEX n = std::min(static_cast<ROWINDEX>(nVisRows), nNextRows);
- m_Dib.SetBlendMode(true);
- DrawPatternData(hdc, nNextPat, false, false,
- 0, n, xofs, rcClient, &ypaint);
- m_Dib.SetBlendMode(false);
- }
- }
- }
- // Drawing outside pattern area
- xpaint = m_szHeader.cx + (ncols - xofs) * nColumnWidth;
- if ((xpaint < rcClient.right) && (ypaint > rcClient.top))
- {
- rc.SetRect(xpaint, rcClient.top, rcClient.right, ypaint);
- ::SetDCBrushColor(hdc, faceColor);
- ::FillRect(hdc, &rc, dcBrush);
- }
- if (ypaint < rcClient.bottom)
- {
- int width = Util::ScalePixels(1, m_hWnd);
- rc.SetRect(0, ypaint, rcClient.right + 1, rcClient.bottom + 1);
- if(width == 1)
- DrawButtonRect(hdc, &rc, _T(""));
- else
- DrawEdge(hdc, rc, EDGE_RAISED, BF_TOPLEFT | BF_MIDDLE); // Prevent lower edge from being drawn
- }
- // Drawing pattern selection
- if(m_Status[psDragnDropping])
- {
- DrawDragSel(hdc);
- }
- const auto buttonBrush = GetSysColorBrush(COLOR_BTNFACE), blackBrush = GetStockBrush(BLACK_BRUSH);
- UINT ncolhdr = xofs;
- xpaint = m_szHeader.cx;
- ypaint = rcClient.top;
- rect.SetRect(0, rcClient.top, rcClient.right, rcClient.top + m_szHeader.cy);
- if(::RectVisible(hdc, &rect))
- {
- sprintf(s, "#%u", m_nPattern);
- rect.right = m_szHeader.cx;
- DrawButtonRect(hdc, &rect, s, FALSE,
- (m_bInItemRect && m_nDragItem.Type() == DragItem::PatternHeader) ? TRUE : FALSE);
- const int dropWidth = Util::ScalePixels(2, m_hWnd);
- // Drawing Channel Headers
- while (xpaint < rcClient.right)
- {
- rect.SetRect(xpaint, ypaint, xpaint + nColumnWidth, ypaint + m_szHeader.cy);
- if (ncolhdr < ncols)
- {
- const auto &channel = sndFile.ChnSettings[ncolhdr];
- const auto recordGroup = pModDoc->GetChannelRecordGroup(static_cast<CHANNELINDEX>(ncolhdr));
- const char *pszfmt = sndFile.m_bChannelMuteTogglePending[ncolhdr]? "[Channel %u]" : "Channel %u";
- if(channel.szName[0] != 0)
- pszfmt = sndFile.m_bChannelMuteTogglePending[ncolhdr] ? "%u: [%s]" : "%u: %s";
- else if(m_nDetailLevel < PatternCursor::volumeColumn)
- pszfmt = sndFile.m_bChannelMuteTogglePending[ncolhdr] ? "[Ch%u]" : "Ch%u";
- else if(m_nDetailLevel < PatternCursor::effectColumn)
- pszfmt = sndFile.m_bChannelMuteTogglePending[ncolhdr] ? "[Chn %u]" : "Chn %u";
- sprintf(s, pszfmt, ncolhdr + 1, channel.szName.buf);
- DrawButtonRect(hdc, &rect, s,
- channel.dwFlags[CHN_MUTE] ? TRUE : FALSE,
- (m_bInItemRect && m_nDragItem.Type() == DragItem::ChannelHeader && m_nDragItem.Value() == ncolhdr) ? TRUE : FALSE,
- recordGroup != RecordGroup::NoGroup ? DT_RIGHT : DT_CENTER, chanColorHeight);
- if(channel.color != ModChannelSettings::INVALID_COLOR)
- {
- // Channel color
- CRect r;
- r.top = rect.top + chanColorOffset;
- r.bottom = r.top + chanColorHeight;
- r.left = rect.left + chanColorOffset;
- r.right = rect.right - chanColorOffset;
- ::SetDCBrushColor(hdc, channel.color);
- ::FillRect(hdc, r, dcBrush);
- }
- // When dragging around channel headers, mark insertion position
- if(m_Status[psDragging] && !m_bInItemRect
- && m_nDragItem.Type() == DragItem::ChannelHeader
- && m_nDropItem.Type() == DragItem::ChannelHeader
- && m_nDropItem.Value() == ncolhdr)
- {
- CRect r;
- r.top = rect.top;
- r.bottom = rect.bottom;
- // Drop position depends on whether hovered channel is left or right of dragged item.
- r.left = (m_nDropItem.Value() < m_nDragItem.Value() || m_Status[psShiftDragging]) ? rect.left : rect.right - dropWidth;
- r.right = r.left + dropWidth;
- ::SetDCBrushColor(hdc, textColor);
- ::FillRect(hdc, r, dcBrush);
- }
- rect.bottom = rect.top + colHeight;
- rect.top += chanColorHeight;
- if(recordGroup != RecordGroup::NoGroup)
- {
- CRect insRect;
- insRect.SetRect(xpaint, ypaint + chanColorHeight, xpaint + nColumnWidth / 8 + recordInsX, ypaint + colHeight);
- FrameRect(hdc, &rect, buttonBrush);
- InvertRect(hdc, &rect);
- s[0] = (recordGroup == RecordGroup::Group1) ? '1' : '2';
- s[1] = '\0';
- DrawButtonRect(hdc, &insRect, s, FALSE, FALSE, DT_CENTER);
- FrameRect(hdc, &insRect, blackBrush);
- }
- if(m_Status[psShowVUMeters])
- {
- OldVUMeters[ncolhdr] = 0;
- DrawChannelVUMeter(hdc, rect.left + 1, rect.bottom, ncolhdr);
- rect.top += vuHeight;
- rect.bottom += vuHeight;
- }
- if(m_Status[psShowPluginNames])
- {
- rect.top = rect.bottom;
- rect.bottom = rect.top + m_szPluginHeader.cy;
- if(PLUGINDEX mixPlug = channel.nMixPlugin; mixPlug != 0)
- sprintf(s, "%u: %s", mixPlug, (sndFile.m_MixPlugins[mixPlug - 1]).pMixPlugin ? sndFile.m_MixPlugins[mixPlug - 1].GetNameLocale() : "[empty]");
- else
- sprintf(s, "---");
- DrawButtonRect(hdc, &rect, s, channel.dwFlags[CHN_NOFX] ? TRUE : FALSE,
- ((m_bInItemRect) && (m_nDragItem.Type() == DragItem::PluginName) && (m_nDragItem.Value() == ncolhdr)) ? TRUE : FALSE, DT_CENTER);
- }
- } else break;
- ncolhdr++;
- xpaint += nColumnWidth;
- }
- }
- if(doSmoothScroll)
- {
- CRect clipRect;
- pDC->GetClipBox(clipRect);
- pDC->BitBlt(clipRect.left, clipRect.top, clipRect.Width(), clipRect.Height(), &m_offScreenDC, clipRect.left, clipRect.top, SRCCOPY);
- SelectBitmap(m_offScreenDC, oldBitmap);
- }
- //rewbs.fxVis
- if (m_pEffectVis)
- {
- //HACK: Update visualizer on every pattern redraw. Cleary there's space for opt here.
- if (m_pEffectVis->m_hWnd) m_pEffectVis->Update();
- }
- }
- static constexpr UINT EncodeRowColor(int rowBkCol, int rowCol, bool rowSelected)
- {
- return (rowBkCol << 16) | (rowCol << 8) | (rowSelected ? 1 : 0);
- }
- void CViewPattern::DrawPatternData(HDC hdc, PATTERNINDEX nPattern, bool selEnable,
- bool isPlaying, ROWINDEX startRow, ROWINDEX numRows, CHANNELINDEX startChan, CRect &rcClient, int *pypaint)
- {
- uint8 selectedCols[MAX_BASECHANNELS]; // Bit mask of selected channel components
- static_assert(1 << PatternCursor::lastColumn <= Util::MaxValueOfType(selectedCols[0]) , "Columns are used as bitmasks.");
- const CSoundFile &sndFile = GetDocument()->GetSoundFile();
- if(!sndFile.Patterns.IsValidPat(nPattern))
- {
- return;
- }
- const CPattern &pattern = sndFile.Patterns[nPattern];
- const PATTERNFONT *pfnt = PatternFont::currentFont;
- CRect rect;
- int xpaint, ypaint = *pypaint;
- UINT nColumnWidth;
-
- CHANNELINDEX ncols = sndFile.GetNumChannels();
- nColumnWidth = m_szCell.cx;
- rect.SetRect(m_szHeader.cx, rcClient.top, m_szHeader.cx+nColumnWidth, rcClient.bottom);
- for(CHANNELINDEX cmk = startChan; cmk < ncols; cmk++)
- {
- selectedCols[cmk] = selEnable ? m_Selection.GetSelectionBits(cmk) : 0;
- if (!::RectVisible(hdc, &rect)) selectedCols[cmk] |= COLUMN_BITS_INVISIBLE;
- rect.left += nColumnWidth;
- rect.right += nColumnWidth;
- }
- // Max Visible Column
- CHANNELINDEX maxcol = ncols;
- while ((maxcol > startChan) && (selectedCols[maxcol-1] & COLUMN_BITS_INVISIBLE)) maxcol--;
- // Init bitmap border
- {
- UINT maxndx = sndFile.GetNumChannels() * m_szCell.cx;
- UINT ibmp = 0;
- if (maxndx > (UINT)m_Dib.GetWidth()) maxndx = m_Dib.GetWidth();
- do
- {
- ibmp += nColumnWidth;
- m_Dib.TextBlt(ibmp-4, 0, 4, m_szCell.cy, pfnt->nClrX+pfnt->nWidth-4, pfnt->nClrY, pfnt->dib);
- } while (ibmp + nColumnWidth <= maxndx);
- }
-
- const bool hexNumbers = (TrackerSettings::Instance().m_dwPatternSetup & PATTERN_HEXDISPLAY);
- bool bRowSel = false;
- int row_col = -1, row_bkcol = -1;
- for(ROWINDEX row = startRow; row < numRows; row++)
- {
- UINT col, xbmp, nbmp, oldrowcolor;
- const int compRow = row + TrackerSettings::Instance().rowDisplayOffset;
- rect.left = 0;
- rect.top = ypaint;
- rect.right = rcClient.right;
- rect.bottom = rect.top + m_szCell.cy;
- if (!::RectVisible(hdc, &rect))
- {
- // No speedup for these columns next time
- for(CHANNELINDEX iup = startChan; iup < maxcol; iup++)
- selectedCols[iup] &= ~COLUMN_BITS_SKIP;
- // skip row
- ypaint += m_szCell.cy;
- if(ypaint >= rcClient.bottom)
- break;
- continue;
- }
- rect.right = rect.left + m_szHeader.cx;
- bool rowDisabled = sndFile.m_lockRowStart != ROWINDEX_INVALID && (row < sndFile.m_lockRowStart || row > sndFile.m_lockRowEnd);
- TCHAR s[32];
- if(hexNumbers)
- wsprintf(s, _T("%s%02X"), compRow < 0 ? _T("-") : _T(""), std::abs(compRow));
- else
- wsprintf(s, _T("%d"), compRow);
- DrawButtonRect(hdc, &rect, s, !selEnable || rowDisabled);
- oldrowcolor = EncodeRowColor(row_bkcol, row_col, bRowSel);
- bRowSel = (m_Selection.ContainsVertical(PatternCursor(row)));
- row_col = MODCOLOR_TEXTNORMAL;
- row_bkcol = MODCOLOR_BACKNORMAL;
- // time signature highlighting
- ROWINDEX nBeat = sndFile.m_nDefaultRowsPerBeat, nMeasure = sndFile.m_nDefaultRowsPerMeasure;
- if(sndFile.Patterns[nPattern].GetOverrideSignature())
- {
- nBeat = sndFile.Patterns[nPattern].GetRowsPerBeat();
- nMeasure = sndFile.Patterns[nPattern].GetRowsPerMeasure();
- }
- // secondary highlight (beats)
- ROWINDEX highlightRow = compRow;
- if(nMeasure > 0)
- highlightRow %= nMeasure;
- if ((TrackerSettings::Instance().m_dwPatternSetup & PATTERN_2NDHIGHLIGHT)
- && nBeat > 0)
- {
- if((highlightRow % nBeat) == 0)
- {
- row_bkcol = MODCOLOR_2NDHIGHLIGHT;
- }
- }
- // primary highlight (measures)
- if((TrackerSettings::Instance().m_dwPatternSetup & PATTERN_STDHIGHLIGHT)
- && nMeasure > 0)
- {
- if(highlightRow == 0)
- {
- row_bkcol = MODCOLOR_BACKHILIGHT;
- }
- }
- bool blendModeChanged = false;
- if (selEnable)
- {
- if ((row == m_nPlayRow) && (nPattern == m_nPlayPat))
- {
- row_col = MODCOLOR_TEXTPLAYCURSOR;
- row_bkcol = MODCOLOR_BACKPLAYCURSOR;
- }
- if (row == GetCurrentRow())
- {
- if(m_Status[psFocussed])
- {
- row_col = MODCOLOR_TEXTCURROW;
- row_bkcol = MODCOLOR_BACKCURROW;
- } else
- if(m_Status[psFollowSong] && isPlaying)
- {
- row_col = MODCOLOR_TEXTPLAYCURSOR;
- row_bkcol = MODCOLOR_BACKPLAYCURSOR;
- }
- }
- blendModeChanged = (rowDisabled != m_Dib.GetBlendMode());
- m_Dib.SetBlendMode(rowDisabled);
- }
- // Eliminate non-visible column
- xpaint = m_szHeader.cx;
- col = startChan;
- while ((selectedCols[col] & COLUMN_BITS_INVISIBLE) && (col < maxcol))
- {
- selectedCols[col] &= ~COLUMN_BITS_SKIP;
- col++;
- xpaint += nColumnWidth;
- }
- // Optimization: same row color ?
- bool useSpeedUpMask = (oldrowcolor == EncodeRowColor(row_bkcol, row_col, bRowSel)) && !blendModeChanged;
- xbmp = nbmp = 0;
- do
- {
- int x, bk_col, tx_col, col_sel, fx_col;
- const ModCommand *m = pattern.GetpModCommand(row, static_cast<CHANNELINDEX>(col));
- // Should empty volume commands be replaced with a volume command showing the default volume?
- const bool drawDefaultVolume = DrawDefaultVolume(m);
- DWORD dwSpeedUpMask = 0;
- if (useSpeedUpMask && (selectedCols[col] & COLUMN_BITS_SKIP) && (row))
- {
- const ModCommand *mold = m - ncols;
- const bool drawOldDefaultVolume = DrawDefaultVolume(mold);
- if (m->note == mold->note) dwSpeedUpMask |= COLUMN_BITS_NOTE;
- if ((m->instr == mold->instr) || (m_nDetailLevel < PatternCursor::instrColumn)) dwSpeedUpMask |= COLUMN_BITS_INSTRUMENT;
- if ( m->IsPcNote() || mold->IsPcNote() )
- {
- // Handle speedup mask for PC notes.
- if(m->note == mold->note)
- {
- if(m->GetValueVolCol() == mold->GetValueVolCol() || (m_nDetailLevel < PatternCursor::volumeColumn)) dwSpeedUpMask |= COLUMN_BITS_VOLUME;
- if(m->GetValueEffectCol() == mold->GetValueEffectCol() || (m_nDetailLevel < PatternCursor::effectColumn)) dwSpeedUpMask |= COLUMN_BITS_FXCMDANDPARAM;
- }
- } else
- {
- if ((m->volcmd == mold->volcmd && (m->volcmd == VOLCMD_NONE || m->vol == mold->vol) && !drawDefaultVolume && !drawOldDefaultVolume) || (m_nDetailLevel < PatternCursor::volumeColumn)) dwSpeedUpMask |= COLUMN_BITS_VOLUME;
- if ((m->command == mold->command) || (m_nDetailLevel < PatternCursor::effectColumn)) dwSpeedUpMask |= (m->command != CMD_NONE) ? COLUMN_BITS_FXCMD : COLUMN_BITS_FXCMDANDPARAM;
- }
- if (dwSpeedUpMask == COLUMN_BITS_ALLCOLUMNS) goto DoBlit;
- }
- selectedCols[col] |= COLUMN_BITS_SKIP;
- col_sel = 0;
- if (bRowSel) col_sel = selectedCols[col] & COLUMN_BITS_ALL;
- tx_col = row_col;
- bk_col = row_bkcol;
- if (col_sel)
- {
- tx_col = MODCOLOR_TEXTSELECTED;
- bk_col = MODCOLOR_BACKSELECTED;
- }
- // Speedup: Empty command which is either not or fully selected
- if (m->IsEmpty() && ((!col_sel) || (col_sel == COLUMN_BITS_ALLCOLUMNS)))
- {
- m_Dib.SetTextColor(tx_col, bk_col);
- m_Dib.TextBlt(xbmp, 0, nColumnWidth-4, m_szCell.cy, pfnt->nClrX, pfnt->nClrY, pfnt->dib);
- goto DoBlit;
- }
- x = 0;
- // Note
- if (!(dwSpeedUpMask & COLUMN_BITS_NOTE))
- {
- tx_col = row_col;
- bk_col = row_bkcol;
- if((TrackerSettings::Instance().m_dwPatternSetup & PATTERN_EFFECTHILIGHT) && m->IsNote())
- {
- tx_col = MODCOLOR_NOTE;
- if(sndFile.m_SongFlags[SONG_AMIGALIMITS | SONG_PT_MODE])
- {
- // Highlight notes that exceed the Amiga's frequency range.
- if(sndFile.GetType() == MOD_TYPE_MOD && (m->note < NOTE_MIDDLEC - 12 || m->note >= NOTE_MIDDLEC + 2 * 12))
- {
- tx_col = MODCOLOR_DODGY_COMMANDS;
- } else if(sndFile.GetType() == MOD_TYPE_S3M && m->instr != 0 && m->instr <= sndFile.GetNumSamples())
- {
- uint32 period = sndFile.GetPeriodFromNote(m->note, 0, sndFile.GetSample(m->instr).nC5Speed);
- if(period < 113 * 4 || period > 856 * 4)
- {
- tx_col = MODCOLOR_DODGY_COMMANDS;
- }
- }
- }
- }
- if (col_sel & COLUMN_BITS_NOTE)
- {
- tx_col = MODCOLOR_TEXTSELECTED;
- bk_col = MODCOLOR_BACKSELECTED;
- }
- // Drawing note
- m_Dib.SetTextColor(tx_col, bk_col);
- if(sndFile.GetType() == MOD_TYPE_MPT && m->instr < MAX_INSTRUMENTS && sndFile.Instruments[m->instr])
- DrawNote(xbmp+x, 0, m->note, sndFile.Instruments[m->instr]->pTuning);
- else //Original
- DrawNote(xbmp+x, 0, m->note);
- }
- x += pfnt->nEltWidths[0];
- // Instrument
- if (m_nDetailLevel >= PatternCursor::instrColumn)
- {
- if (!(dwSpeedUpMask & COLUMN_BITS_INSTRUMENT))
- {
- tx_col = row_col;
- bk_col = row_bkcol;
- if ((TrackerSettings::Instance().m_dwPatternSetup & PATTERN_EFFECTHILIGHT) && (m->instr))
- {
- tx_col = MODCOLOR_INSTRUMENT;
- }
- if (col_sel & COLUMN_BITS_INSTRUMENT)
- {
- tx_col = MODCOLOR_TEXTSELECTED;
- bk_col = MODCOLOR_BACKSELECTED;
- }
- // Drawing instrument
- m_Dib.SetTextColor(tx_col, bk_col);
- DrawInstrument(xbmp+x, 0, m->instr);
- }
- x += pfnt->nEltWidths[1];
- }
- // Volume
- if (m_nDetailLevel >= PatternCursor::volumeColumn)
- {
- if (!(dwSpeedUpMask & COLUMN_BITS_VOLUME))
- {
- tx_col = row_col;
- bk_col = row_bkcol;
- if (col_sel & COLUMN_BITS_VOLUME)
- {
- tx_col = MODCOLOR_TEXTSELECTED;
- bk_col = MODCOLOR_BACKSELECTED;
- } else if (!m->IsPcNote() && (TrackerSettings::Instance().m_dwPatternSetup & PATTERN_EFFECTHILIGHT))
- {
- if(m->volcmd != VOLCMD_NONE && m->volcmd < MAX_VOLCMDS && effectColors[m->GetVolumeEffectType()] != 0)
- {
- tx_col = effectColors[m->GetVolumeEffectType()];
- } else if(drawDefaultVolume)
- {
- tx_col = MODCOLOR_DEFAULTVOLUME;
- }
- }
- // Drawing Volume
- m_Dib.SetTextColor(tx_col, bk_col);
- DrawVolumeCommand(xbmp + x, 0, *m, drawDefaultVolume);
- }
- x += pfnt->nEltWidths[2];
- }
- // Command & param
- if (m_nDetailLevel >= PatternCursor::effectColumn)
- {
- const bool isPCnote = m->IsPcNote();
- uint16 val = m->GetValueEffectCol();
- if(val > ModCommand::maxColumnValue) val = ModCommand::maxColumnValue;
- fx_col = row_col;
- if (!isPCnote && m->command != CMD_NONE && m->command < MAX_EFFECTS && (TrackerSettings::Instance().m_dwPatternSetup & PATTERN_EFFECTHILIGHT))
- {
- if(effectColors[m->GetEffectType()] != 0)
- fx_col = effectColors[m->GetEffectType()];
- else if(m->command == CMD_DUMMY)
- fx_col = MODCOLOR_DUMMYCOMMAND;
- }
- if (!(dwSpeedUpMask & COLUMN_BITS_FXCMD))
- {
- tx_col = fx_col;
- bk_col = row_bkcol;
- if (col_sel & COLUMN_BITS_FXCMD)
- {
- tx_col = MODCOLOR_TEXTSELECTED;
- bk_col = MODCOLOR_BACKSELECTED;
- }
- // Drawing Command
- m_Dib.SetTextColor(tx_col, bk_col);
- if(isPCnote)
- {
- m_Dib.TextBlt(xbmp + x, 0, 2, pfnt->spacingY, pfnt->nClrX+x, pfnt->nClrY, pfnt->dib);
- m_Dib.TextBlt(xbmp + x + pfnt->pcValMargin, 0, pfnt->nEltWidths[3], m_szCell.cy, pfnt->nNumX, pfnt->nNumY+(val / 100)*pfnt->spacingY, pfnt->dib);
- } else
- {
- if(m->command != CMD_NONE)
- {
- char n = sndFile.GetModSpecifications().GetEffectLetter(m->command);
- MPT_ASSERT(n >= ' ');
- DrawLetter(xbmp+x, 0, n, pfnt->nEltWidths[3], pfnt->nCmdOfs);
- } else
- {
- m_Dib.TextBlt(xbmp+x, 0, pfnt->nEltWidths[3], pfnt->spacingY, pfnt->nClrX+x, pfnt->nClrY, pfnt->dib);
- }
- }
- DrawPadding(m_Dib, pfnt, xbmp + x, 0, 3);
- }
- x += pfnt->nEltWidths[3];
- // Param
- if (!(dwSpeedUpMask & COLUMN_BITS_FXPARAM))
- {
- tx_col = fx_col;
- bk_col = row_bkcol;
- if (col_sel & COLUMN_BITS_FXPARAM)
- {
- tx_col = MODCOLOR_TEXTSELECTED;
- bk_col = MODCOLOR_BACKSELECTED;
- }
- // Drawing param
- m_Dib.SetTextColor(tx_col, bk_col);
- if(isPCnote)
- {
- m_Dib.TextBlt(xbmp + x, 0, pfnt->nParamHiWidth, m_szCell.cy, pfnt->nNumX, pfnt->nNumY+((val / 10) % 10)*pfnt->spacingY, pfnt->dib);
- m_Dib.TextBlt(xbmp + x + pfnt->nParamHiWidth, 0, pfnt->nEltWidths[4] - pfnt->padding[4] - pfnt->nParamHiWidth, m_szCell.cy, pfnt->nNumX+pfnt->paramLoMargin, pfnt->nNumY+(val % 10)*pfnt->spacingY, pfnt->dib);
- }
- else
- {
- if (m->command)
- {
- m_Dib.TextBlt(xbmp + x, 0, pfnt->nParamHiWidth, m_szCell.cy, pfnt->nNumX, pfnt->nNumY+(m->param >> 4)*pfnt->spacingY, pfnt->dib);
- m_Dib.TextBlt(xbmp + x + pfnt->nParamHiWidth, 0, pfnt->nEltWidths[4] - pfnt->padding[4] - pfnt->nParamHiWidth, m_szCell.cy, pfnt->nNumX+pfnt->paramLoMargin, pfnt->nNumY+(m->param & 0x0F)*pfnt->spacingY, pfnt->dib);
- } else
- {
- m_Dib.TextBlt(xbmp+x, 0, pfnt->nEltWidths[4], m_szCell.cy, pfnt->nClrX+x, pfnt->nClrY, pfnt->dib);
- }
- }
- DrawPadding(m_Dib, pfnt, xbmp + x, 0, 4);
- }
- }
- DoBlit:
- nbmp++;
- xbmp += nColumnWidth;
- xpaint += nColumnWidth;
- if ((xbmp + nColumnWidth >= (UINT)m_Dib.GetWidth()) || (xpaint >= rcClient.right)) break;
- } while (++col < maxcol);
- m_Dib.Blit(hdc, xpaint-xbmp, ypaint, xbmp, m_szCell.cy);
- // skip row
- ypaint += m_szCell.cy;
- if (ypaint >= rcClient.bottom) break;
- }
- *pypaint = ypaint;
- }
- void CViewPattern::DrawChannelVUMeter(HDC hdc, int x, int y, UINT nChn)
- {
- if (ChnVUMeters[nChn] != OldVUMeters[nChn])
- {
- UINT vul, vur;
- vul = (ChnVUMeters[nChn] & 0xFF00) >> 8;
- vur = ChnVUMeters[nChn] & 0xFF;
- vul /= 15;
- vur /= 15;
- if (vul > 8) vul = 8;
- if (vur > 8) vur = 8;
- x += (m_szCell.cx / 2);
- const auto &channel = GetSoundFile()->m_PlayState.Chn[nChn];
- const bool isSynth =
- channel.dwFlags[CHN_ADLIB]
- || (channel.pModSample != nullptr && channel.pModSample->uFlags[CHN_ADLIB])
- || ((channel.pModSample == nullptr || !channel.pModSample->HasSampleData()) && channel.HasMIDIOutput());
- const auto bmp = isSynth ? CMainFrame::bmpPluginVUMeters : CMainFrame::bmpVUMeters;
- if (m_nDetailLevel <= PatternCursor::instrColumn)
- {
- DibBlt(hdc, x-VUMETERS_LOWIDTH-1, y, VUMETERS_LOWIDTH, VUMETERS_BMPHEIGHT,
- VUMETERS_BMPWIDTH*2+VUMETERS_MEDWIDTH*2, vul * VUMETERS_BMPHEIGHT, bmp);
- DibBlt(hdc, x-1, y, VUMETERS_LOWIDTH, VUMETERS_BMPHEIGHT,
- VUMETERS_BMPWIDTH*2+VUMETERS_MEDWIDTH*2+VUMETERS_LOWIDTH, vur * VUMETERS_BMPHEIGHT, bmp);
- } else
- if (m_nDetailLevel <= PatternCursor::volumeColumn)
- {
- DibBlt(hdc, x - VUMETERS_MEDWIDTH-1, y, VUMETERS_MEDWIDTH, VUMETERS_BMPHEIGHT,
- VUMETERS_BMPWIDTH*2, vul * VUMETERS_BMPHEIGHT, bmp);
- DibBlt(hdc, x, y, VUMETERS_MEDWIDTH, VUMETERS_BMPHEIGHT,
- VUMETERS_BMPWIDTH*2+VUMETERS_MEDWIDTH, vur * VUMETERS_BMPHEIGHT, bmp);
- } else
- {
- DibBlt(hdc, x - VUMETERS_BMPWIDTH - 1, y, VUMETERS_BMPWIDTH, VUMETERS_BMPHEIGHT,
- 0, vul * VUMETERS_BMPHEIGHT, bmp);
- DibBlt(hdc, x + 1, y, VUMETERS_BMPWIDTH, VUMETERS_BMPHEIGHT,
- VUMETERS_BMPWIDTH, vur * VUMETERS_BMPHEIGHT, bmp);
- }
- OldVUMeters[nChn] = ChnVUMeters[nChn];
- }
- }
- // Draw an inverted border around the dragged selection.
- void CViewPattern::DrawDragSel(HDC hdc)
- {
- const CSoundFile *pSndFile = GetSoundFile();
- CRect rect;
- int x1, y1, x2, y2;
- int nChannels, nRows;
- if(pSndFile == nullptr || !pSndFile->Patterns.IsValidPat(m_nPattern)) return;
- // Compute relative movement
- int dx = (int)m_DragPos.GetChannel() - (int)m_StartSel.GetChannel();
- int dy = (int)m_DragPos.GetRow() - (int)m_StartSel.GetRow();
- // Compute destination rect
- PatternCursor begin(m_Selection.GetUpperLeft()), end(m_Selection.GetLowerRight());
- // Check which selection lines need to be drawn.
- bool drawLeft = ((int)begin.GetChannel() + dx >= GetXScrollPos());
- bool drawRight = ((int)end.GetChannel() + dx < (int)pSndFile->GetNumChannels());
- bool drawTop = ((int)begin.GetRow() + dy >= GetYScrollPos() - (int)m_nMidRow);
- bool drawBottom = ((int)end.GetRow() + dy < (int)pSndFile->Patterns[m_nPattern].GetNumRows());
- begin.Move(dy, dx, 0);
- if(begin.GetChannel() >= pSndFile->GetNumChannels())
- {
- // Moved outside pattern range.
- return;
- }
- end.Move(dy, dx, 0);
- begin.Sanitize(pSndFile->Patterns[m_nPattern].GetNumRows(), pSndFile->GetNumChannels());
- end.Sanitize(pSndFile->Patterns[m_nPattern].GetNumRows(), pSndFile->GetNumChannels());
- // We need to know the first pixel that's not part of our rect anymore, so we extend the selection.
- end.Move(1, 0, 1);
- PatternRect destination(begin, end);
- x1 = m_Selection.GetStartChannel();
- y1 = m_Selection.GetStartRow();
- x2 = m_Selection.GetEndChannel();
- y2 = m_Selection.GetEndRow();
- PatternCursor::Columns c1 = m_Selection.GetStartColumn();
- PatternCursor::Columns c2 = m_Selection.GetEndColumn();
- x1 += dx;
- x2 += dx;
- y1 += dy;
- y2 += dy;
- nChannels = pSndFile->m_nChannels;
- nRows = pSndFile->Patterns[m_nPattern].GetNumRows();
- if (x1 < GetXScrollPos()) drawLeft = false;
- if (x1 >= nChannels) x1 = nChannels - 1;
- if (x1 < 0) { x1 = 0; c1 = PatternCursor::firstColumn; drawLeft = false; }
- if (x2 >= nChannels) { x2 = nChannels - 1; c2 = PatternCursor::lastColumn; drawRight = false; }
- if (x2 < 0) x2 = 0;
- if (y1 < GetYScrollPos() - (int)m_nMidRow) drawTop = false;
- if (y1 >= nRows) y1 = nRows-1;
- if (y1 < 0) { y1 = 0; drawTop = false; }
- if (y2 >= nRows) { y2 = nRows-1; drawBottom = false; }
- if (y2 < 0) y2 = 0;
- POINT ptTopLeft = GetPointFromPosition(begin);
- POINT ptBottomRight = GetPointFromPosition(end);
- if ((ptTopLeft.x >= ptBottomRight.x) || (ptTopLeft.y >= ptBottomRight.y)) return;
- if(end.GetColumnType() == PatternCursor::firstColumn)
- {
- // Special case: If selection ends on the last column of a channel, subtract the channel separator width.
- ptBottomRight.x -= 4;
- }
- // invert the brush pattern (looks just like frame window sizing)
- ::SetTextColor(hdc, RGB(255, 255, 255));
- ::SetBkColor(hdc, RGB(0, 0, 0));
- CBrush* pBrush = CDC::GetHalftoneBrush();
- if (pBrush != NULL)
- {
- HBRUSH hOldBrush = (HBRUSH)SelectObject(hdc, pBrush->m_hObject);
- // Top
- if (drawTop)
- {
- rect.SetRect(ptTopLeft.x + 4, ptTopLeft.y, ptBottomRight.x, ptTopLeft.y + 4);
- if (!drawLeft) rect.left -= 4;
- PatBlt(hdc, rect.left, rect.top, rect.Width(), rect.Height(), PATINVERT);
- }
- // Bottom
- if (drawBottom)
- {
- rect.SetRect(ptTopLeft.x, ptBottomRight.y - 4, ptBottomRight.x - 4, ptBottomRight.y);
- if (!drawRight) rect.right += 4;
- PatBlt(hdc, rect.left, rect.top, rect.Width(), rect.Height(), PATINVERT);
- }
- // Left
- if (drawLeft)
- {
- rect.SetRect(ptTopLeft.x, ptTopLeft.y, ptTopLeft.x + 4, ptBottomRight.y - 4);
- if (!drawBottom) rect.bottom += 4;
- PatBlt(hdc, rect.left, rect.top, rect.Width(), rect.Height(), PATINVERT);
- }
- // Right
- if (drawRight)
- {
- rect.SetRect(ptBottomRight.x - 4, ptTopLeft.y + 4, ptBottomRight.x, ptBottomRight.y);
- if (!drawTop) rect.top -= 4;
- PatBlt(hdc, rect.left, rect.top, rect.Width(), rect.Height(), PATINVERT);
- }
- if (hOldBrush != NULL) SelectObject(hdc, hOldBrush);
- }
- }
- void CViewPattern::OnDrawDragSel()
- {
- HDC hdc = ::GetDC(m_hWnd);
- if (hdc != NULL)
- {
- DrawDragSel(hdc);
- ::ReleaseDC(m_hWnd, hdc);
- }
- }
- ///////////////////////////////////////////////////////////////////////////////
- // CViewPattern Scrolling Functions
- void CViewPattern::UpdateScrollSize()
- {
- const CSoundFile *pSndFile = GetSoundFile();
- const CHANNELINDEX numChannels = pSndFile ? pSndFile->GetNumChannels() : 0;
- const ROWINDEX numRows = (pSndFile && pSndFile->Patterns.IsValidPat(m_nPattern)) ? pSndFile->Patterns[m_nPattern].GetNumRows() : 0;
- CRect rect;
- SIZE sizeTotal, sizePage, sizeLine;
- sizeTotal.cx = m_szHeader.cx + numChannels * m_szCell.cx;
- sizeTotal.cy = m_szHeader.cy + numRows * m_szCell.cy;
- sizeLine.cx = m_szCell.cx;
- sizeLine.cy = m_szCell.cy;
- sizePage.cx = sizeLine.cx * 2;
- sizePage.cy = sizeLine.cy * 8;
- GetClientRect(&rect);
- m_nMidRow = 0;
- if (TrackerSettings::Instance().m_dwPatternSetup & PATTERN_CENTERROW) m_nMidRow = (rect.Height() - m_szHeader.cy) / (m_szCell.cy * 2);
- if (m_nMidRow) sizeTotal.cy += m_nMidRow * m_szCell.cy * 2;
- SetScrollSizes(MM_TEXT, sizeTotal, sizePage, sizeLine);
- m_bWholePatternFitsOnScreen = (rect.Height() >= sizeTotal.cy);
- if(m_bWholePatternFitsOnScreen)
- m_nYScroll = 0;
- }
- void CViewPattern::UpdateScrollPos()
- {
- CRect rect;
- GetClientRect(&rect);
-
- int x = GetScrollPos(SB_HORZ);
- if (x < 0) x = 0;
- m_nXScroll = (x + m_szCell.cx - 1) / m_szCell.cx;
- int y = GetScrollPos(SB_VERT);
- if (y < 0) y = 0;
- m_nYScroll = (y + m_szCell.cy - 1) / m_szCell.cy;
- }
- BOOL CViewPattern::OnScrollBy(CSize sizeScroll, BOOL bDoScroll)
- {
- int xOrig, xNew, x;
- int yOrig, yNew, y;
- // don't scroll if there is no valid scroll range (ie. no scroll bar)
- CScrollBar* pBar;
- DWORD dwStyle = GetStyle();
- pBar = GetScrollBarCtrl(SB_VERT);
- if ((pBar != NULL && !pBar->IsWindowEnabled()) ||
- (pBar == NULL && !(dwStyle & WS_VSCROLL)))
- {
- // vertical scroll bar not enabled
- sizeScroll.cy = 0;
- }
- pBar = GetScrollBarCtrl(SB_HORZ);
- if ((pBar != NULL && !pBar->IsWindowEnabled()) ||
- (pBar == NULL && !(dwStyle & WS_HSCROLL)))
- {
- // horizontal scroll bar not enabled
- sizeScroll.cx = 0;
- }
- // adjust current x position
- xOrig = x = GetScrollPos(SB_HORZ);
- int xMax = GetScrollLimit(SB_HORZ);
- x += sizeScroll.cx;
- if (x < 0) x = 0; else if (x > xMax) x = xMax;
- // adjust current y position
- yOrig = y = GetScrollPos(SB_VERT);
- int yMax = GetScrollLimit(SB_VERT);
- y += sizeScroll.cy;
- if (y < 0) y = 0; else if (y > yMax) y = yMax;
- if (!bDoScroll) return TRUE;
- xNew = x;
- yNew = y;
- if (x > 0) x = (x + m_szCell.cx - 1) / m_szCell.cx; else x = 0;
- if (y > 0) y = (y + m_szCell.cy - 1) / m_szCell.cy; else y = 0;
- if ((x != m_nXScroll) || (y != m_nYScroll))
- {
- CRect rect;
- GetClientRect(&rect);
- // HACK:
- // Wine handles ScrollWindow completely synchronously (using RedrawWindow).
- // This causes the window update region to be repainted immediately
- // before and immediately after the actual copying of the scrolled rect.
- // Async and sync window painting generally do not mix well at all
- // (not even on native Windows) and this causes inevitable flickering
- // on Wine.
- // Instead, just invalidate the whole scrolled window area and let
- // WM_PAINT handle the whole mess without ever scrolling any already
- // painted contents. This causes additional CPU usage (on Wine) but
- // avoids totally annoying and distracting flickering of the current-row-
- // highlight.
- if (x != m_nXScroll)
- {
- rect.left = m_szHeader.cx;
- rect.top = 0;
- if(TrackerSettings::Instance().patternAlwaysDrawWholePatternOnScrollSlow || mpt::OS::Windows::IsWine())
- {
- InvalidateRect(&rect, FALSE);
- } else
- {
- ScrollWindow((m_nXScroll - x) * GetChannelWidth(), 0, &rect, &rect);
- }
- m_nXScroll = x;
- }
- if (y != m_nYScroll)
- {
- rect.left = 0;
- rect.top = m_szHeader.cy;
- if(TrackerSettings::Instance().patternAlwaysDrawWholePatternOnScrollSlow || mpt::OS::Windows::IsWine())
- {
- InvalidateRect(&rect, FALSE);
- } else
- {
- ScrollWindow(0, (m_nYScroll - y) * GetRowHeight(), &rect, &rect);
- }
- m_nYScroll = y;
- }
- }
- if (xNew != xOrig) SetScrollPos(SB_HORZ, xNew);
- if (yNew != yOrig) SetScrollPos(SB_VERT, yNew);
- return TRUE;
- }
- void CViewPattern::OnSize(UINT nType, int cx, int cy)
- {
- // Note: Switching between modules (when MDI childs are maximized) first calls this with the windowed size, then with the maximized size.
- // Watch out for this odd behaviour when debugging this function.
- CScrollView::OnSize(nType, cx, cy);
- if (((nType == SIZE_RESTORED) || (nType == SIZE_MAXIMIZED)) && (cx > 0) && (cy > 0))
- {
- UpdateSizes();
- UpdateScrollSize();
- UpdateScrollPos();
- m_Dib.SetSize(cx + m_szCell.cx, m_szCell.cy);
- InvalidatePattern();
- }
- }
- void CViewPattern::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
- {
- if (nSBCode == SB_THUMBTRACK) m_Status.set(psDragVScroll);
- CModScrollView::OnVScroll(nSBCode, nPos, pScrollBar);
- if (nSBCode == SB_ENDSCROLL) m_Status.reset(psDragVScroll);
- }
- void CViewPattern::SetCurSel(PatternCursor beginSel, PatternCursor endSel)
- {
- RECT rect1, rect2, rect, rcInt, rcUni;
- POINT pt;
- // Get current selection area
- PatternCursor endSel2(m_Selection.GetLowerRight());
- endSel2.Move(1, 0, 1);
- pt = GetPointFromPosition(m_Selection.GetUpperLeft());
- rect1.left = pt.x;
- rect1.top = pt.y;
- pt = GetPointFromPosition(endSel2);
- rect1.right = pt.x;
- rect1.bottom = pt.y;
- if(rect1.left < m_szHeader.cx) rect1.left = m_szHeader.cx;
- if(rect1.top < m_szHeader.cy) rect1.top = m_szHeader.cy;
- // Get new selection area
- m_Selection = PatternRect(beginSel, endSel);
- if(const CSoundFile *sndFile = GetSoundFile(); sndFile != nullptr && sndFile->Patterns.IsValidPat(m_nPattern))
- {
- m_Selection.Sanitize(sndFile->Patterns[m_nPattern].GetNumRows(), sndFile->GetNumChannels());
- }
- UpdateIndicator();
- pt = GetPointFromPosition(m_Selection.GetUpperLeft());
- rect2.left = pt.x;
- rect2.top = pt.y;
- endSel2.Set(m_Selection.GetLowerRight());
- endSel2.Move(1, 0, 1);
- pt = GetPointFromPosition(endSel2);
- rect2.right = pt.x;
- rect2.bottom = pt.y;
- if (rect2.left < m_szHeader.cx) rect2.left = m_szHeader.cx;
- if (rect2.top < m_szHeader.cy) rect2.top = m_szHeader.cy;
- // Compute area for invalidation
- IntersectRect(&rcInt, &rect1, &rect2);
- UnionRect(&rcUni, &rect1, &rect2);
- SubtractRect(&rect, &rcUni, &rcInt);
- if ((rect.left == rcUni.left) && (rect.top == rcUni.top)
- && (rect.right == rcUni.right) && (rect.bottom == rcUni.bottom))
- {
- InvalidateRect(&rect1, FALSE);
- InvalidateRect(&rect2, FALSE);
- } else
- {
- InvalidateRect(&rect, FALSE);
- }
- }
- void CViewPattern::InvalidatePattern(bool invalidateChannelHeaders, bool invalidateRowHeaders)
- {
- CRect rect;
- GetClientRect(&rect);
- if(!invalidateChannelHeaders)
- {
- rect.top += m_szHeader.cy;
- }
- if(!invalidateRowHeaders)
- {
- rect.left += m_szHeader.cx;
- }
- InvalidateRect(&rect, FALSE);
- SanitizeCursor();
- }
- void CViewPattern::InvalidateRow(ROWINDEX n)
- {
- const CSoundFile *pSndFile = GetSoundFile();
- if(pSndFile && pSndFile->Patterns.IsValidPat(m_nPattern))
- {
- int yofs = GetYScrollPos() - m_nMidRow;
- if (n == ROWINDEX_INVALID) n = GetCurrentRow();
- if (((int)n < yofs) || (n >= pSndFile->Patterns[m_nPattern].GetNumRows())) return;
- CRect rect;
- GetClientRect(&rect);
- rect.left = m_szHeader.cx;
- rect.top = m_szHeader.cy - GetSmoothScrollOffset();
- rect.top += (n - yofs) * m_szCell.cy;
- rect.bottom = rect.top + m_szCell.cy;
- InvalidateRect(&rect, FALSE);
- }
- }
- void CViewPattern::InvalidateArea(PatternCursor begin, PatternCursor end)
- {
- RECT rect;
- POINT pt;
- pt = GetPointFromPosition(begin);
- rect.left = pt.x;
- rect.top = pt.y;
- end.Move(1, 0, 1);
- pt = GetPointFromPosition(end);
- rect.right = pt.x;
- rect.bottom = pt.y;
- InvalidateRect(&rect, FALSE);
- }
- void CViewPattern::InvalidateCell(PatternCursor cursor)
- {
- cursor.RemoveColType();
- InvalidateArea(cursor, PatternCursor(cursor.GetRow(), cursor.GetChannel(), PatternCursor::lastColumn));
- }
- void CViewPattern::InvalidateChannelsHeaders(CHANNELINDEX chn)
- {
- CRect rect;
- GetClientRect(&rect);
- rect.bottom = rect.top + m_szHeader.cy;
- if(chn != CHANNELINDEX_INVALID)
- {
- rect.left = GetPointFromPosition(PatternCursor{ 0u, chn }).x;
- rect.right = rect.left + GetChannelWidth();
- }
- InvalidateRect(&rect, FALSE);
- }
- void CViewPattern::UpdateIndicator(bool updateAccessibility)
- {
- const CSoundFile *sndFile = GetSoundFile();
- CMainFrame *mainFrm = CMainFrame::GetMainFrame();
- if(mainFrm == nullptr || sndFile == nullptr || !sndFile->Patterns.IsValidPat(m_nPattern))
- return;
- mainFrm->SetUserText(MPT_CFORMAT("Row {}, Col {}")(GetCurrentRow(), GetCurrentChannel() + 1));
- if(::GetFocus() == m_hWnd)
- {
- const bool hasSelection = m_Selection.GetUpperLeft() != m_Selection.GetLowerRight();
- if(hasSelection)
- mainFrm->SetInfoText(MPT_CFORMAT("Selection: {} row{}, {} channel{}")(m_Selection.GetNumRows(), CString(m_Selection.GetNumRows() != 1 ? _T("s") : _T("")), m_Selection.GetNumChannels(), CString(m_Selection.GetNumChannels() != 1 ? _T("s") : _T(""))));
- if(GetCurrentRow() < sndFile->Patterns[m_nPattern].GetNumRows() && m_Cursor.GetChannel() < sndFile->GetNumChannels())
- {
- if(!hasSelection)
- mainFrm->SetInfoText(GetCursorDescription());
- UpdateXInfoText();
- }
- if(updateAccessibility)
- mainFrm->NotifyAccessibilityUpdate(*this);
- }
- }
- CString CViewPattern::GetCursorDescription() const
- {
- const CSoundFile &sndFile = *GetSoundFile();
- CString s;
- if(!sndFile.Patterns.IsValidPat(m_nPattern))
- {
- return s;
- }
- ROWINDEX row = m_Cursor.GetRow();
- CHANNELINDEX channel = m_Cursor.GetChannel();
- const ModCommand *m = sndFile.Patterns[m_nPattern].GetpModCommand(row, channel);
- switch(m_Cursor.GetColumnType())
- {
- case PatternCursor::noteColumn:
- // display note
- if(m->IsSpecialNote())
- s = szSpecialNoteShortDesc[m->note - NOTE_MIN_SPECIAL];
- else if(m->IsNote())
- s = mpt::ToCString(sndFile.GetNoteName(m->note, m->instr));
- break;
- case PatternCursor::instrColumn:
- // display instrument
- if(m->instr)
- {
- s.Format(_T("%u: "), m->instr);
- if(m->IsPcNote())
- {
- // display plugin name.
- if(m->instr <= MAX_MIXPLUGINS)
- {
- s += mpt::ToCString(sndFile.m_MixPlugins[m->instr - 1].GetName());
- }
- } else
- {
- // "normal" instrument
- if(sndFile.GetNumInstruments())
- {
- if((m->instr <= sndFile.GetNumInstruments()) && (sndFile.Instruments[m->instr]))
- {
- ModInstrument *pIns = sndFile.Instruments[m->instr];
- s += mpt::ToCString(sndFile.GetCharsetInternal(), pIns->name);
- if((m->note) && (m->note <= NOTE_MAX))
- {
- const SAMPLEINDEX nsmp = pIns->Keyboard[m->note - 1];
- if((nsmp) && (nsmp <= sndFile.GetNumSamples()))
- {
- if(sndFile.m_szNames[nsmp][0])
- {
- s.AppendFormat(_T(" (%d: "), nsmp);
- s += mpt::ToCString(sndFile.GetCharsetInternal(), sndFile.m_szNames[nsmp]);
- s.AppendChar(_T(')'));
- }
- }
- }
- }
- } else if(m->instr <= sndFile.GetNumSamples())
- {
- s += mpt::ToCString(sndFile.GetCharsetInternal(), sndFile.m_szNames[m->instr]);
- }
- }
- }
- break;
- case PatternCursor::volumeColumn:
- // display volume command
- if(m->IsPcNote())
- {
- // display plugin param name.
- if(m->instr > 0 && m->instr <= MAX_MIXPLUGINS)
- {
- const SNDMIXPLUGIN &plug = sndFile.m_MixPlugins[m->instr - 1];
- if(plug.pMixPlugin != nullptr)
- {
- s = plug.pMixPlugin->GetFormattedParamName(m->GetValueVolCol());
- }
- }
- } else if(m->volcmd != VOLCMD_NONE)
- {
- // "normal" volume command
- EffectInfo effectInfo(sndFile);
- effectInfo.GetVolCmdInfo(effectInfo.GetIndexFromVolCmd(m->volcmd), &s);
- s += _T(": ");
- CString tmp;
- effectInfo.GetVolCmdParamInfo(*m, &tmp);
- s += tmp;
- }
- break;
- case PatternCursor::effectColumn:
- case PatternCursor::paramColumn:
- // display effect command
- if(m->IsPcNote())
- {
- s.Format(_T("Parameter value: %u"), m->GetValueEffectCol());
- } else if(m->command != CMD_NONE)
- {
- EffectInfo effectInfo(sndFile);
- CString sztmp;
- if(effectInfo.GetIndexFromEffect(m->command, m->param) >= 0)
- {
- UINT xParam = 0, xMultiplier = 1;
- getXParam(m->command, m_nPattern, row, channel, sndFile, xParam, xMultiplier);
- effectInfo.GetEffectNameEx(sztmp, *m, m->param * xMultiplier + xParam, channel);
- }
- //effectInfo.GetEffectName(sztmp, m->command, m->param, false, nChn);
- if(!sztmp.IsEmpty())
- {
- s.Format(_T("%c%02X: "), sndFile.GetModSpecifications().GetEffectLetter(m->command), m->param);
- s += sztmp;
- }
- }
- break;
- }
- return s;
- }
- void CViewPattern::UpdateXInfoText()
- {
- const CSoundFile *sndFile = GetSoundFile();
- CMainFrame *mainFrm = CMainFrame::GetMainFrame();
- if(mainFrm == nullptr || sndFile == nullptr)
- return;
- CHANNELINDEX chn = GetCurrentChannel();
- const auto &channel = sndFile->m_PlayState.Chn[chn];
- CString xtraInfo;
- xtraInfo.Format(_T("Chn:%d; Vol:%X; Mac:%X; Cut:%X%s; Res:%X; Pan:%X%s"),
- chn + 1,
- channel.nGlobalVol,
- channel.nActiveMacro,
- channel.nCutOff,
- (channel.nFilterMode == FilterMode::HighPass) ? _T("-HP") : _T(""),
- channel.nResonance,
- channel.nPan,
- channel.dwFlags[CHN_SURROUND] ? _T("-S") : _T(""));
- mainFrm->SetXInfoText(xtraInfo);
- }
- void CViewPattern::UpdateAllVUMeters(Notification *pnotify)
- {
- CMainFrame *pMainFrm = CMainFrame::GetMainFrame();
- const CModDoc *pModDoc = GetDocument();
-
- if ((!pModDoc) || (!pMainFrm)) return;
- CRect rcClient;
- GetClientRect(&rcClient);
- int xofs = GetXScrollPos();
- HDC hdc = ::GetDC(m_hWnd);
- const bool isPlaying = (pMainFrm->GetFollowSong(pModDoc) == m_hWnd);
- int x = m_szHeader.cx;
- CHANNELINDEX nChn = static_cast<CHANNELINDEX>(xofs);
- const int yPos = rcClient.top + MulDiv(COLHDR_HEIGHT, m_nDPIy, 96);
- while ((nChn < pModDoc->GetNumChannels()) && (x < rcClient.right))
- {
- ChnVUMeters[nChn] = static_cast<uint16>(pnotify->pos[nChn]);
- if ((!isPlaying) || pnotify->type[Notification::Stop]) ChnVUMeters[nChn] = 0;
- DrawChannelVUMeter(hdc, x + 1, rcClient.top + yPos, nChn);
- nChn++;
- x += m_szCell.cx;
- }
- ::ReleaseDC(m_hWnd, hdc);
- }
- OPENMPT_NAMESPACE_END
|