Skip to content

Commit

Permalink
Merge pull request mixxxdj#13685 from ronso0/duplicate-track-location
Browse files Browse the repository at this point in the history
(fix) Track menu: avoid crash and UX issues with track nullptr
  • Loading branch information
daschuer authored Sep 27, 2024
2 parents 9db85cf + 580a8fc commit 19c0e66
Show file tree
Hide file tree
Showing 3 changed files with 54 additions and 24 deletions.
17 changes: 9 additions & 8 deletions src/library/dlgtagfetcher.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -246,30 +246,31 @@ void DlgTagFetcher::slotPrev() {
}

void DlgTagFetcher::loadTrack(const TrackPointer& pTrack) {
tags->clear();
m_data = Data();
if (m_pTrack) {
tags->clear();
disconnect(m_pTrack.get(),
&Track::changed,
this,
&DlgTagFetcher::slotTrackChanged);
m_data = Data();
}
tags->clear();

m_pWFetchedCoverArtLabel->setCoverArt(CoverInfo{}, QPixmap{});

m_coverCache.clear();

m_pTrack = pTrack;
if (!m_pTrack) {
return;
}

btnRetry->setDisabled(true);
btnApply->setDisabled(true);
checkBoxTags->setDisabled(true);
checkBoxCover->setDisabled(true);
statusMessage->setVisible(false);

m_pTrack = pTrack;
if (!m_pTrack) {
loadingProgressBar->setVisible(false);
return;
}

loadingProgressBar->setVisible(true);
loadingProgressBar->setValue(kMinimumValueOfQProgressBar);
loadingProgressBar->setToolTip(QString());
Expand Down
2 changes: 2 additions & 0 deletions src/library/dlgtrackinfo.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -623,6 +623,8 @@ void DlgTrackInfo::saveTrack() {
void DlgTrackInfo::clear() {
const QSignalBlocker signalBlocker(this);

setWindowTitle(QString());

if (m_pLoadedTrack) {
disconnect(m_pLoadedTrack.get(),
&Track::changed,
Expand Down
59 changes: 43 additions & 16 deletions src/widget/wtrackmenu.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -215,6 +215,10 @@ void WTrackMenu::createMenus() {
m_pSearchRelatedMenu->clear();
const auto pTrack = getFirstTrackPointer();
if (pTrack) {
// Ensure it's enabled, else we can't add actions.
VERIFY_OR_DEBUG_ASSERT(m_pSearchRelatedMenu->isEnabled()) {
m_pSearchRelatedMenu->setEnabled(true);
}
m_pSearchRelatedMenu->addActionsForTrack(*pTrack);
}
m_pSearchRelatedMenu->setEnabled(
Expand Down Expand Up @@ -722,11 +726,9 @@ std::pair<bool, bool> WTrackMenu::getTrackBpmLockStates() const {
break;
}
}
} else {
if (m_pTrack) {
anyBpmLocked = m_pTrack->isBpmLocked();
anyBpmNotLocked = !anyBpmLocked;
}
} else if (m_pTrack) {
anyBpmLocked = m_pTrack->isBpmLocked();
anyBpmNotLocked = !anyBpmLocked;
}
return std::pair<bool, bool>(anyBpmLocked, anyBpmNotLocked);
}
Expand Down Expand Up @@ -815,8 +817,11 @@ CoverInfo WTrackMenu::getCoverInfoOfLastTrack() const {
.data()
.toString();
return coverInfo;
} else {
} else if (m_pTrack) {
return m_pTrack->getCoverInfoWithLocation();
} else {
// No track, no track model
return CoverInfo();
}
}

Expand All @@ -828,10 +833,25 @@ void WTrackMenu::updateMenus() {
// Gray out some stuff if multiple songs were selected.
const bool singleTrackSelected = getTrackCount() == 1;

if (featureIsEnabled(Feature::SearchRelated)) {
// Enable only if we have one valid track pointer.
// this prevents the cursor getting stuck on this menu in case it gets
// disabled when encountering a track nullptr in lambda function
// connected to aboutToShow() signal (see createMenus()).
// Note: track nullptr can happen when TrackDAO returns nullptr because
// the selected track references a file referenced by another cached track.
DEBUG_ASSERT(m_pSearchRelatedMenu);
const auto pTrack = getFirstTrackPointer();
m_pSearchRelatedMenu->setEnabled(pTrack != nullptr);
// TODO Only enable for single track?
}

if (featureIsEnabled(Feature::LoadTo)) {
// Enable menus only for single track
int iNumDecks = static_cast<int>(m_pNumDecks.get());
m_pDeckMenu->clear();
if (iNumDecks > 0) {
m_pDeckMenu->setEnabled(singleTrackSelected);
if (singleTrackSelected && iNumDecks > 0) {
for (int i = 1; i <= iNumDecks; ++i) {
// PlayerManager::groupForDeck is 0-indexed.
QString deckGroup = PlayerManager::groupForDeck(i - 1);
Expand Down Expand Up @@ -866,8 +886,9 @@ void WTrackMenu::updateMenus() {

int iNumSamplers = static_cast<int>(m_pNumSamplers.get());
const int maxSamplersPerMenu = 16;
if (iNumSamplers > 0) {
m_pSamplerMenu->clear();
m_pSamplerMenu->clear();
m_pSamplerMenu->setEnabled(singleTrackSelected);
if (singleTrackSelected && iNumSamplers > 0) {
QMenu* pMenu = m_pSamplerMenu;
int samplersInMenu = 0;
for (int i = 1; i <= iNumSamplers; ++i) {
Expand Down Expand Up @@ -968,13 +989,15 @@ void WTrackMenu::updateMenus() {
} else if (m_pTrack) {
pTrack = m_pTrack;
}
const double bpm = pTrack->getBpm();
appendBpmPreviewtoBpmAction(m_pBpmDoubleAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmHalveAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmTwoThirdsAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmThreeFourthsAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmFourThirdsAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmThreeHalvesAction, bpm);
if (pTrack) {
const double bpm = pTrack->getBpm();
appendBpmPreviewtoBpmAction(m_pBpmDoubleAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmHalveAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmTwoThirdsAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmThreeFourthsAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmFourThirdsAction, bpm);
appendBpmPreviewtoBpmAction(m_pBpmThreeHalvesAction, bpm);
}
}
}
}
Expand Down Expand Up @@ -2335,6 +2358,10 @@ void WTrackMenu::slotShowDlgTrackInfo() {
}
});
// Method getFirstTrackPointer() is not applicable here!
// DlgTrackInfo relies on a track model for certain operations,
// for example show/hide the Next/Prev buttons.
// It can be loaded with either an index (must have a model),
// or a TrackPointer (must NOT have a model then).
if (m_pTrackModel) {
m_pDlgTrackInfo->loadTrack(m_trackIndexList.at(0));
} else {
Expand Down

0 comments on commit 19c0e66

Please sign in to comment.