/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "msgCore.h" #include "nsMsgQuickSearchDBView.h" #include "nsMsgFolderFlags.h" #include "nsIMsgHdr.h" #include "nsIMsgImapMailFolder.h" #include "nsImapCore.h" #include "nsIMsgHdr.h" #include "nsIDBFolderInfo.h" #include "nsMsgMessageFlags.h" #include "nsMsgUtils.h" nsMsgQuickSearchDBView::nsMsgQuickSearchDBView() { m_usingCachedHits = false; m_cacheEmpty = true; } nsMsgQuickSearchDBView::~nsMsgQuickSearchDBView() {} NS_IMPL_ISUPPORTS_INHERITED(nsMsgQuickSearchDBView, nsMsgDBView, nsIMsgDBView, nsIMsgSearchNotify) NS_IMETHODIMP nsMsgQuickSearchDBView::Open(nsIMsgFolder* folder, nsMsgViewSortTypeValue sortType, nsMsgViewSortOrderValue sortOrder, nsMsgViewFlagsTypeValue viewFlags, int32_t* pCount) { nsresult rv = nsMsgDBView::Open(folder, sortType, sortOrder, viewFlags, pCount); NS_ENSURE_SUCCESS(rv, rv); if (!m_db) return NS_ERROR_NULL_POINTER; m_viewFolder = nullptr; int32_t count; rv = InitThreadedView(count); if (pCount) *pCount = count; return rv; } NS_IMETHODIMP nsMsgQuickSearchDBView::CloneDBView(nsIMessenger* aMessengerInstance, nsIMsgWindow* aMsgWindow, nsIMsgDBViewCommandUpdater* aCmdUpdater, nsIMsgDBView** _retval) { nsMsgQuickSearchDBView* newMsgDBView = new nsMsgQuickSearchDBView(); nsresult rv = CopyDBView(newMsgDBView, aMessengerInstance, aMsgWindow, aCmdUpdater); NS_ENSURE_SUCCESS(rv, rv); NS_IF_ADDREF(*_retval = newMsgDBView); return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::CopyDBView(nsMsgDBView* aNewMsgDBView, nsIMessenger* aMessengerInstance, nsIMsgWindow* aMsgWindow, nsIMsgDBViewCommandUpdater* aCmdUpdater) { nsMsgThreadedDBView::CopyDBView(aNewMsgDBView, aMessengerInstance, aMsgWindow, aCmdUpdater); nsMsgQuickSearchDBView* newMsgDBView = (nsMsgQuickSearchDBView*)aNewMsgDBView; // now copy all of our private member data newMsgDBView->m_origKeys = m_origKeys.Clone(); return NS_OK; } nsresult nsMsgQuickSearchDBView::DeleteMessages( nsIMsgWindow* window, nsTArray const& selection, bool deleteStorage) { for (nsMsgViewIndex viewIndex : selection) { nsCOMPtr msgHdr; (void)GetMsgHdrForViewIndex(viewIndex, getter_AddRefs(msgHdr)); if (msgHdr) { RememberDeletedMsgHdr(msgHdr); } } return nsMsgDBView::DeleteMessages(window, selection, deleteStorage); } NS_IMETHODIMP nsMsgQuickSearchDBView::DoCommand( nsMsgViewCommandTypeValue aCommand) { if (aCommand == nsMsgViewCommandType::markAllRead) { nsresult rv = NS_OK; m_folder->EnableNotifications(nsIMsgFolder::allMessageCountNotifications, false); for (uint32_t i = 0; NS_SUCCEEDED(rv) && i < GetSize(); i++) { nsCOMPtr msgHdr; m_db->GetMsgHdrForKey(m_keys[i], getter_AddRefs(msgHdr)); rv = m_db->MarkHdrRead(msgHdr, true, nullptr); } m_folder->EnableNotifications(nsIMsgFolder::allMessageCountNotifications, true); nsCOMPtr imapFolder = do_QueryInterface(m_folder); if (NS_SUCCEEDED(rv) && imapFolder) rv = imapFolder->StoreImapFlags(kImapMsgSeenFlag, true, m_keys, nullptr); m_db->SetSummaryValid(true); return rv; } else return nsMsgDBView::DoCommand(aCommand); } NS_IMETHODIMP nsMsgQuickSearchDBView::GetViewType( nsMsgViewTypeValue* aViewType) { NS_ENSURE_ARG_POINTER(aViewType); *aViewType = nsMsgViewType::eShowQuickSearchResults; return NS_OK; } nsresult nsMsgQuickSearchDBView::AddHdr(nsIMsgDBHdr* msgHdr, nsMsgViewIndex* resultIndex) { nsMsgKey msgKey; msgHdr->GetMessageKey(&msgKey); // protect against duplication. if (m_origKeys.BinaryIndexOf(msgKey) == m_origKeys.NoIndex) { nsMsgViewIndex insertIndex = GetInsertIndexHelper( msgHdr, m_origKeys, nullptr, nsMsgViewSortOrder::ascending, nsMsgViewSortType::byId); m_origKeys.InsertElementAt(insertIndex, msgKey); } if (m_viewFlags & (nsMsgViewFlagsType::kGroupBySort | nsMsgViewFlagsType::kThreadedDisplay)) { nsMsgKey parentKey; msgHdr->GetThreadParent(&parentKey); return nsMsgThreadedDBView::OnNewHeader(msgHdr, parentKey, true); } else return nsMsgDBView::AddHdr(msgHdr, resultIndex); } nsresult nsMsgQuickSearchDBView::OnNewHeader(nsIMsgDBHdr* newHdr, nsMsgKey aParentKey, bool ensureListed) { if (newHdr) { bool match = false; nsCOMPtr searchSession = do_QueryReferent(m_searchSession); if (searchSession) searchSession->MatchHdr(newHdr, m_db, &match); if (match) { // put the new header in m_origKeys, so that expanding a thread will // show the newly added header. nsMsgKey newKey; (void)newHdr->GetMessageKey(&newKey); nsMsgViewIndex insertIndex = GetInsertIndexHelper( newHdr, m_origKeys, nullptr, nsMsgViewSortOrder::ascending, nsMsgViewSortType::byId); m_origKeys.InsertElementAt(insertIndex, newKey); nsMsgThreadedDBView::OnNewHeader( newHdr, aParentKey, ensureListed); // do not add a new message if there isn't a match. } } return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::OnHdrFlagsChanged( nsIMsgDBHdr* aHdrChanged, uint32_t aOldFlags, uint32_t aNewFlags, nsIDBChangeListener* aInstigator) { nsresult rv = nsMsgGroupView::OnHdrFlagsChanged(aHdrChanged, aOldFlags, aNewFlags, aInstigator); if (m_viewFolder && (m_viewFolder != m_folder) && (aOldFlags & nsMsgMessageFlags::Read) != (aNewFlags & nsMsgMessageFlags::Read)) { // if we're displaying a single folder virtual folder for an imap folder, // the search criteria might be on message body, and we might not have the // message body offline, in which case we can't tell if the message // matched or not. But if the unread flag changed, we need to update the // unread counts. Normally, VirtualFolderChangeListener::OnHdrFlagsChanged // will handle this, but it won't work for body criteria when we don't have // the body offline. nsCOMPtr imapFolder = do_QueryInterface(m_viewFolder); if (imapFolder) { nsMsgViewIndex hdrIndex = FindHdr(aHdrChanged); if (hdrIndex != nsMsgViewIndex_None) { nsCOMPtr searchSession = do_QueryReferent(m_searchSession); if (searchSession) { bool oldMatch, newMatch; rv = searchSession->MatchHdr(aHdrChanged, m_db, &newMatch); aHdrChanged->SetFlags(aOldFlags); rv = searchSession->MatchHdr(aHdrChanged, m_db, &oldMatch); aHdrChanged->SetFlags(aNewFlags); // if it doesn't match the criteria, // VirtualFolderChangeListener::OnHdrFlagsChanged won't tweak the // read/unread counts. So do it here: if (!oldMatch && !newMatch) { nsCOMPtr virtDatabase; nsCOMPtr dbFolderInfo; rv = m_viewFolder->GetDBFolderInfoAndDB( getter_AddRefs(dbFolderInfo), getter_AddRefs(virtDatabase)); NS_ENSURE_SUCCESS(rv, rv); dbFolderInfo->ChangeNumUnreadMessages( (aOldFlags & nsMsgMessageFlags::Read) ? 1 : -1); m_viewFolder->UpdateSummaryTotals(true); // force update from db. virtDatabase->Commit(nsMsgDBCommitType::kLargeCommit); } } } } } return rv; } NS_IMETHODIMP nsMsgQuickSearchDBView::OnHdrPropertyChanged(nsIMsgDBHdr* aHdrChanged, const nsACString& property, bool aPreChange, uint32_t* aStatus, nsIDBChangeListener* aInstigator) { // If the junk mail plugin just activated on a message, then // we'll allow filters to remove from view. // Otherwise, just update the view line. // // Note this will not add newly matched headers to the view. This is // probably a bug that needs fixing. NS_ENSURE_ARG_POINTER(aStatus); NS_ENSURE_ARG_POINTER(aHdrChanged); nsMsgViewIndex index = FindHdr(aHdrChanged); if (index == nsMsgViewIndex_None) // message does not appear in view return NS_OK; nsCString originStr; (void)aHdrChanged->GetStringProperty("junkscoreorigin", originStr); // check for "plugin" with only first character for performance bool plugin = (originStr.get()[0] == 'p'); if (aPreChange) { // first call, done prior to the change *aStatus = plugin; return NS_OK; } // second call, done after the change bool wasPlugin = *aStatus; bool match = true; nsCOMPtr searchSession( do_QueryReferent(m_searchSession)); if (searchSession) searchSession->MatchHdr(aHdrChanged, m_db, &match); if (!match && plugin && !wasPlugin) RemoveByIndex(index); // remove hdr from view else NoteChange(index, 1, nsMsgViewNotificationCode::changed); return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::GetSearchSession(nsIMsgSearchSession** aSession) { NS_ASSERTION(false, "GetSearchSession method is not implemented"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsMsgQuickSearchDBView::SetSearchSession(nsIMsgSearchSession* aSession) { m_searchSession = do_GetWeakReference(aSession); return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::OnSearchHit(nsIMsgDBHdr* aMsgHdr, nsIMsgFolder* folder) { NS_ENSURE_ARG(aMsgHdr); if (!m_db) return NS_ERROR_NULL_POINTER; // remember search hit and when search is done, reconcile cache // with new hits; m_hdrHits.AppendObject(aMsgHdr); nsMsgKey key; aMsgHdr->GetMessageKey(&key); // Is FindKey going to be expensive here? A lot of hits could make // it a little bit slow to search through the view for every hit. if (m_cacheEmpty || FindKey(key, false) == nsMsgViewIndex_None) return AddHdr(aMsgHdr); else return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::OnSearchDone(nsresult status) { // This batch began in OnNewSearch. if (mJSTree) mJSTree->EndUpdateBatch(); // We're a single-folder virtual folder if viewFolder != folder, and that is // the only case in which we want to be messing about with a results cache // or unread counts. if (m_db && m_viewFolder && m_viewFolder != m_folder) { nsTArray keyArray; nsCString searchUri; m_viewFolder->GetURI(searchUri); uint32_t count = m_hdrHits.Count(); // Build up message keys. The cache expects them to be sorted. for (uint32_t i = 0; i < count; i++) { nsMsgKey key; m_hdrHits[i]->GetMessageKey(&key); keyArray.AppendElement(key); } keyArray.Sort(); nsTArray staleHits; nsresult rv = m_db->RefreshCache(searchUri, keyArray, staleHits); NS_ENSURE_SUCCESS(rv, rv); for (nsMsgKey staleKey : staleHits) { nsCOMPtr hdrDeleted; m_db->GetMsgHdrForKey(staleKey, getter_AddRefs(hdrDeleted)); if (hdrDeleted) OnHdrDeleted(hdrDeleted, nsMsgKey_None, 0, this); } nsCOMPtr virtDatabase; nsCOMPtr dbFolderInfo; rv = m_viewFolder->GetDBFolderInfoAndDB(getter_AddRefs(dbFolderInfo), getter_AddRefs(virtDatabase)); NS_ENSURE_SUCCESS(rv, rv); uint32_t numUnread = 0; size_t numTotal = m_origKeys.Length(); for (size_t i = 0; i < m_origKeys.Length(); i++) { bool isRead; m_db->IsRead(m_origKeys[i], &isRead); if (!isRead) numUnread++; } dbFolderInfo->SetNumUnreadMessages(numUnread); dbFolderInfo->SetNumMessages(numTotal); m_viewFolder->UpdateSummaryTotals(true); // force update from db. virtDatabase->Commit(nsMsgDBCommitType::kLargeCommit); } if (m_sortType != nsMsgViewSortType::byThread) // we do not find levels for the results. { m_sortValid = false; // sort the results Sort(m_sortType, m_sortOrder); } if (m_viewFolder && (m_viewFolder != m_folder)) SetMRUTimeForFolder(m_viewFolder); return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::OnNewSearch() { int32_t oldSize = GetSize(); m_keys.Clear(); m_levels.Clear(); m_flags.Clear(); m_hdrHits.Clear(); // this needs to happen after we remove all the keys, since RowCountChanged() // will call our GetRowCount() if (mTree) mTree->RowCountChanged(0, -oldSize); uint32_t folderFlags = 0; if (m_viewFolder) m_viewFolder->GetFlags(&folderFlags); // check if it's a virtual folder - if so, we should get the cached hits // from the db, and set a flag saying that we're using cached values. if (folderFlags & nsMsgFolderFlags::Virtual) { nsCOMPtr cachedHits; nsCString searchUri; m_viewFolder->GetURI(searchUri); m_db->GetCachedHits(searchUri, getter_AddRefs(cachedHits)); if (cachedHits) { bool hasMore; m_usingCachedHits = true; cachedHits->HasMoreElements(&hasMore); m_cacheEmpty = !hasMore; if (mTree) mTree->BeginUpdateBatch(); if (mJSTree) mJSTree->BeginUpdateBatch(); while (hasMore) { nsCOMPtr header; nsresult rv = cachedHits->GetNext(getter_AddRefs(header)); if (header && NS_SUCCEEDED(rv)) AddHdr(header); else break; cachedHits->HasMoreElements(&hasMore); } if (mTree) mTree->EndUpdateBatch(); if (mJSTree) mJSTree->EndUpdateBatch(); } } // Prevent updates for every message found. This batch ends in OnSearchDone. if (mJSTree) mJSTree->BeginUpdateBatch(); return NS_OK; } nsresult nsMsgQuickSearchDBView::GetFirstMessageHdrToDisplayInThread( nsIMsgThread* threadHdr, nsIMsgDBHdr** result) { uint32_t numChildren; nsresult rv = NS_OK; uint8_t minLevel = 0xff; threadHdr->GetNumChildren(&numChildren); nsMsgKey threadRootKey; nsCOMPtr rootParent; threadHdr->GetRootHdr(getter_AddRefs(rootParent)); if (rootParent) rootParent->GetMessageKey(&threadRootKey); else threadHdr->GetThreadKey(&threadRootKey); nsCOMPtr retHdr; // iterate over thread, finding mgsHdr in view with the lowest level. for (uint32_t childIndex = 0; childIndex < numChildren; childIndex++) { nsCOMPtr child; rv = threadHdr->GetChildHdrAt(childIndex, getter_AddRefs(child)); if (NS_SUCCEEDED(rv) && child) { nsMsgKey msgKey; child->GetMessageKey(&msgKey); // this works because we've already sorted m_keys by id. nsMsgViewIndex keyIndex = m_origKeys.BinaryIndexOf(msgKey); if (keyIndex != nsMsgViewIndex_None) { // this is the root, so it's the best we're going to do. if (msgKey == threadRootKey) { retHdr = child; break; } uint8_t level = 0; nsMsgKey parentId; child->GetThreadParent(&parentId); nsCOMPtr parent; // count number of ancestors - that's our level while (parentId != nsMsgKey_None) { m_db->GetMsgHdrForKey(parentId, getter_AddRefs(parent)); if (parent) { nsMsgKey saveParentId = parentId; parent->GetThreadParent(&parentId); // message is it's own parent - bad, let's break out of here. // Or we've got some circular ancestry. if (parentId == saveParentId || level > numChildren) break; level++; } else // if we can't find the parent, don't loop forever. break; } if (level < minLevel) { minLevel = level; retHdr = child; } } } } retHdr.forget(result); return NS_OK; } nsresult nsMsgQuickSearchDBView::SortThreads( nsMsgViewSortTypeValue sortType, nsMsgViewSortOrderValue sortOrder) { // don't need to sort by threads for group view. if (m_viewFlags & nsMsgViewFlagsType::kGroupBySort) return NS_OK; // iterate over the messages in the view, getting the thread id's // sort m_keys so we can quickly find if a key is in the view. m_keys.Sort(); // array of the threads' root hdr keys. nsTArray threadRootIds; nsCOMPtr rootHdr; nsCOMPtr msgHdr; nsCOMPtr threadHdr; for (uint32_t i = 0; i < m_keys.Length(); i++) { GetMsgHdrForViewIndex(i, getter_AddRefs(msgHdr)); m_db->GetThreadContainingMsgHdr(msgHdr, getter_AddRefs(threadHdr)); if (threadHdr) { nsMsgKey rootKey; threadHdr->GetChildKeyAt(0, &rootKey); nsMsgViewIndex threadRootIndex = threadRootIds.BinaryIndexOf(rootKey); // if we already have that id in top level threads, ignore this msg. if (threadRootIndex != nsMsgViewIndex_None) continue; // it would be nice if GetInsertIndexHelper always found the hdr, but it // doesn't. threadHdr->GetChildHdrAt(0, getter_AddRefs(rootHdr)); if (!rootHdr) continue; threadRootIndex = GetInsertIndexHelper(rootHdr, threadRootIds, nullptr, nsMsgViewSortOrder::ascending, nsMsgViewSortType::byId); threadRootIds.InsertElementAt(threadRootIndex, rootKey); } } m_sortType = nsMsgViewSortType::byNone; // sort from scratch // need to sort the top level threads now by sort order, if it's not by id // and ascending (which is the order per above). if (!(sortType == nsMsgViewSortType::byId && sortOrder == nsMsgViewSortOrder::ascending)) { m_keys.SwapElements(threadRootIds); nsMsgDBView::Sort(sortType, sortOrder); threadRootIds.SwapElements(m_keys); } m_keys.Clear(); m_levels.Clear(); m_flags.Clear(); // now we've build up the list of thread ids - need to build the view // from that. So for each thread id, we need to list the messages in the // thread. uint32_t numThreads = threadRootIds.Length(); for (uint32_t threadIndex = 0; threadIndex < numThreads; threadIndex++) { m_db->GetMsgHdrForKey(threadRootIds[threadIndex], getter_AddRefs(rootHdr)); if (rootHdr) { nsCOMPtr displayRootHdr; m_db->GetThreadContainingMsgHdr(rootHdr, getter_AddRefs(threadHdr)); if (threadHdr) { nsMsgKey rootKey; uint32_t rootFlags; GetFirstMessageHdrToDisplayInThread(threadHdr, getter_AddRefs(displayRootHdr)); if (!displayRootHdr) continue; displayRootHdr->GetMessageKey(&rootKey); displayRootHdr->GetFlags(&rootFlags); rootFlags |= MSG_VIEW_FLAG_ISTHREAD; m_keys.AppendElement(rootKey); m_flags.AppendElement(rootFlags); m_levels.AppendElement(0); nsMsgViewIndex startOfThreadViewIndex = m_keys.Length(); nsMsgViewIndex rootIndex = startOfThreadViewIndex - 1; uint32_t numListed = 0; ListIdsInThreadOrder(threadHdr, rootKey, 1, &startOfThreadViewIndex, &numListed); if (numListed > 0) m_flags[rootIndex] = rootFlags | MSG_VIEW_FLAG_HASCHILDREN; } } } // The thread state is left expanded (despite viewFlags) so at least reflect // the correct state. m_viewFlags |= nsMsgViewFlagsType::kExpandAll; return NS_OK; } nsresult nsMsgQuickSearchDBView::ListCollapsedChildren( nsMsgViewIndex viewIndex, nsTArray>& messageArray) { nsCOMPtr threadHdr; nsresult rv = GetThreadContainingIndex(viewIndex, getter_AddRefs(threadHdr)); NS_ENSURE_SUCCESS(rv, rv); uint32_t numChildren; threadHdr->GetNumChildren(&numChildren); nsCOMPtr rootHdr; nsMsgKey rootKey; GetMsgHdrForViewIndex(viewIndex, getter_AddRefs(rootHdr)); rootHdr->GetMessageKey(&rootKey); // group threads can have the root key twice, one for the dummy row. bool rootKeySkipped = false; for (uint32_t i = 0; i < numChildren; i++) { nsCOMPtr msgHdr; threadHdr->GetChildHdrAt(i, getter_AddRefs(msgHdr)); if (msgHdr) { nsMsgKey msgKey; msgHdr->GetMessageKey(&msgKey); if (msgKey != rootKey || (GroupViewUsesDummyRow() && rootKeySkipped)) { // if this hdr is in the original view, add it to new view. if (m_origKeys.BinaryIndexOf(msgKey) != m_origKeys.NoIndex) messageArray.AppendElement(msgHdr); } else { rootKeySkipped = true; } } } return NS_OK; } nsresult nsMsgQuickSearchDBView::ListIdsInThread( nsIMsgThread* threadHdr, nsMsgViewIndex startOfThreadViewIndex, uint32_t* pNumListed) { if (m_viewFlags & nsMsgViewFlagsType::kThreadedDisplay && !(m_viewFlags & nsMsgViewFlagsType::kGroupBySort)) { nsMsgKey parentKey = m_keys[startOfThreadViewIndex++]; return ListIdsInThreadOrder(threadHdr, parentKey, 1, &startOfThreadViewIndex, pNumListed); } uint32_t numChildren; threadHdr->GetNumChildren(&numChildren); uint32_t i; uint32_t viewIndex = startOfThreadViewIndex + 1; nsCOMPtr rootHdr; nsMsgKey rootKey; uint32_t rootFlags = m_flags[startOfThreadViewIndex]; *pNumListed = 0; GetMsgHdrForViewIndex(startOfThreadViewIndex, getter_AddRefs(rootHdr)); rootHdr->GetMessageKey(&rootKey); // group threads can have the root key twice, one for the dummy row. bool rootKeySkipped = false; for (i = 0; i < numChildren; i++) { nsCOMPtr msgHdr; threadHdr->GetChildHdrAt(i, getter_AddRefs(msgHdr)); if (msgHdr != nullptr) { nsMsgKey msgKey; msgHdr->GetMessageKey(&msgKey); if (msgKey != rootKey || (GroupViewUsesDummyRow() && rootKeySkipped)) { nsMsgViewIndex threadRootIndex = m_origKeys.BinaryIndexOf(msgKey); // if this hdr is in the original view, add it to new view. if (threadRootIndex != nsMsgViewIndex_None) { uint32_t childFlags; msgHdr->GetFlags(&childFlags); InsertMsgHdrAt( viewIndex, msgHdr, msgKey, childFlags, FindLevelInThread(msgHdr, startOfThreadViewIndex, viewIndex)); if (!(rootFlags & MSG_VIEW_FLAG_HASCHILDREN)) m_flags[startOfThreadViewIndex] = rootFlags | MSG_VIEW_FLAG_HASCHILDREN; viewIndex++; (*pNumListed)++; } } else { rootKeySkipped = true; } } } return NS_OK; } nsresult nsMsgQuickSearchDBView::ListIdsInThreadOrder( nsIMsgThread* threadHdr, nsMsgKey parentKey, uint32_t level, uint32_t callLevel, nsMsgKey keyToSkip, nsMsgViewIndex* viewIndex, uint32_t* pNumListed) { nsCOMPtr msgEnumerator; nsresult rv = threadHdr->EnumerateMessages(parentKey, getter_AddRefs(msgEnumerator)); NS_ENSURE_SUCCESS(rv, rv); // We use the numChildren as a sanity check on the thread structure. uint32_t numChildren; (void)threadHdr->GetNumChildren(&numChildren); bool hasMore; while (NS_SUCCEEDED(rv = msgEnumerator->HasMoreElements(&hasMore)) && hasMore) { nsCOMPtr msgHdr; rv = msgEnumerator->GetNext(getter_AddRefs(msgHdr)); NS_ENSURE_SUCCESS(rv, rv); nsMsgKey msgKey; msgHdr->GetMessageKey(&msgKey); if (msgKey == keyToSkip) continue; // If we discover depths of more than numChildren, it means we have // some sort of circular thread relationship and we bail out of the // while loop before overflowing the stack with recursive calls. // Technically, this is an error, but forcing a database rebuild // is too destructive so we just return. if (*pNumListed > numChildren || callLevel > numChildren) { NS_ERROR("loop in message threading while listing children"); return NS_OK; } int32_t childLevel = level; if (m_origKeys.BinaryIndexOf(msgKey) != m_origKeys.NoIndex) { uint32_t msgFlags; msgHdr->GetFlags(&msgFlags); InsertMsgHdrAt(*viewIndex, msgHdr, msgKey, msgFlags & ~MSG_VIEW_FLAGS, level); (*pNumListed)++; (*viewIndex)++; childLevel++; } rv = ListIdsInThreadOrder(threadHdr, msgKey, childLevel, callLevel + 1, keyToSkip, viewIndex, pNumListed); NS_ENSURE_SUCCESS(rv, rv); } return rv; } nsresult nsMsgQuickSearchDBView::ListIdsInThreadOrder(nsIMsgThread* threadHdr, nsMsgKey parentKey, uint32_t level, nsMsgViewIndex* viewIndex, uint32_t* pNumListed) { nsresult rv = ListIdsInThreadOrder(threadHdr, parentKey, level, level, nsMsgKey_None, viewIndex, pNumListed); // Because a quick search view might not have the actual thread root // as its root, and thus might have a message that potentially has siblings // as its root, and the enumerator will miss the siblings, we might need to // make a pass looking for the siblings of the non-root root. We'll put // those after the potential children of the root. So we will list the // children of the faux root's parent, ignoring the faux root. if (level == 1) { nsCOMPtr root; nsCOMPtr rootParent; nsMsgKey rootKey; threadHdr->GetRootHdr(getter_AddRefs(rootParent)); if (rootParent) { rootParent->GetMessageKey(&rootKey); if (rootKey != parentKey) rv = ListIdsInThreadOrder(threadHdr, rootKey, level, level, parentKey, viewIndex, pNumListed); } } return rv; } nsresult nsMsgQuickSearchDBView::ExpansionDelta(nsMsgViewIndex index, int32_t* expansionDelta) { *expansionDelta = 0; if (index >= ((nsMsgViewIndex)m_keys.Length())) return NS_MSG_MESSAGE_NOT_FOUND; char flags = m_flags[index]; if (!(m_viewFlags & nsMsgViewFlagsType::kThreadedDisplay)) return NS_OK; nsCOMPtr threadHdr; nsresult rv = GetThreadContainingIndex(index, getter_AddRefs(threadHdr)); NS_ENSURE_SUCCESS(rv, rv); uint32_t numChildren; threadHdr->GetNumChildren(&numChildren); nsCOMPtr rootHdr; nsMsgKey rootKey; GetMsgHdrForViewIndex(index, getter_AddRefs(rootHdr)); rootHdr->GetMessageKey(&rootKey); // group threads can have the root key twice, one for the dummy row. bool rootKeySkipped = false; for (uint32_t i = 0; i < numChildren; i++) { nsCOMPtr msgHdr; threadHdr->GetChildHdrAt(i, getter_AddRefs(msgHdr)); if (msgHdr) { nsMsgKey msgKey; msgHdr->GetMessageKey(&msgKey); if (msgKey != rootKey || (GroupViewUsesDummyRow() && rootKeySkipped)) { // if this hdr is in the original view, add it to new view. if (m_origKeys.BinaryIndexOf(msgKey) != m_origKeys.NoIndex) (*expansionDelta)++; } else { rootKeySkipped = true; } } } if (!(flags & nsMsgMessageFlags::Elided)) *expansionDelta = -(*expansionDelta); return NS_OK; } NS_IMETHODIMP nsMsgQuickSearchDBView::OpenWithHdrs(nsIMsgEnumerator* aHeaders, nsMsgViewSortTypeValue aSortType, nsMsgViewSortOrderValue aSortOrder, nsMsgViewFlagsTypeValue aViewFlags, int32_t* aCount) { if (aViewFlags & nsMsgViewFlagsType::kGroupBySort) return nsMsgGroupView::OpenWithHdrs(aHeaders, aSortType, aSortOrder, aViewFlags, aCount); m_sortType = aSortType; m_sortOrder = aSortOrder; m_viewFlags = aViewFlags; bool hasMore; nsCOMPtr supports; nsresult rv = NS_OK; while (NS_SUCCEEDED(rv = aHeaders->HasMoreElements(&hasMore)) && hasMore) { nsCOMPtr msgHdr; rv = aHeaders->GetNext(getter_AddRefs(msgHdr)); if (NS_SUCCEEDED(rv) && msgHdr) { AddHdr(msgHdr); } else { break; } } *aCount = m_keys.Length(); return rv; } NS_IMETHODIMP nsMsgQuickSearchDBView::SetViewFlags( nsMsgViewFlagsTypeValue aViewFlags) { nsresult rv = NS_OK; // if the grouping has changed, rebuild the view if ((m_viewFlags & nsMsgViewFlagsType::kGroupBySort) ^ (aViewFlags & nsMsgViewFlagsType::kGroupBySort)) rv = RebuildView(aViewFlags); nsMsgDBView::SetViewFlags(aViewFlags); return rv; } nsresult nsMsgQuickSearchDBView::GetMessageEnumerator( nsIMsgEnumerator** enumerator) { return GetViewEnumerator(enumerator); } NS_IMETHODIMP nsMsgQuickSearchDBView::OnHdrDeleted(nsIMsgDBHdr* aHdrDeleted, nsMsgKey aParentKey, int32_t aFlags, nsIDBChangeListener* aInstigator) { NS_ENSURE_ARG_POINTER(aHdrDeleted); nsMsgKey msgKey; aHdrDeleted->GetMessageKey(&msgKey); size_t keyIndex = m_origKeys.BinaryIndexOf(msgKey); if (keyIndex != m_origKeys.NoIndex) m_origKeys.RemoveElementAt(keyIndex); return nsMsgThreadedDBView::OnHdrDeleted(aHdrDeleted, aParentKey, aFlags, aInstigator); } NS_IMETHODIMP nsMsgQuickSearchDBView::GetNumMsgsInView(int32_t* aNumMsgs) { NS_ENSURE_ARG_POINTER(aNumMsgs); *aNumMsgs = m_origKeys.Length(); return NS_OK; }