mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-14 15:37:55 +00:00
349 lines
12 KiB
C++
349 lines
12 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
// vim:cindent:ai:sw=4:ts=4:et:
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is nsCounterManager.
|
|
*
|
|
* The Initial Developer of the Original Code is the Mozilla Foundation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2004
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* L. David Baron <dbaron@dbaron.org> (original author)
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
/* implementation of CSS counters (for numbering things) */
|
|
|
|
#include "nsCounterManager.h"
|
|
#include "nsBulletFrame.h" // legacy location for list style type to text code
|
|
#include "nsContentUtils.h"
|
|
|
|
// assign the correct |mValueAfter| value to a node that has been inserted
|
|
// Should be called immediately after calling |Insert|.
|
|
void nsCounterUseNode::Calc(nsCounterList *aList)
|
|
{
|
|
NS_ASSERTION(!aList->IsDirty(),
|
|
"Why are we calculating with a dirty list?");
|
|
mValueAfter = aList->ValueBefore(this);
|
|
}
|
|
|
|
// assign the correct |mValueAfter| value to a node that has been inserted
|
|
// Should be called immediately after calling |Insert|.
|
|
void nsCounterChangeNode::Calc(nsCounterList *aList)
|
|
{
|
|
NS_ASSERTION(!aList->IsDirty(),
|
|
"Why are we calculating with a dirty list?");
|
|
if (mType == RESET) {
|
|
mValueAfter = mChangeValue;
|
|
} else {
|
|
NS_ASSERTION(mType == INCREMENT, "invalid type");
|
|
mValueAfter = aList->ValueBefore(this) + mChangeValue;
|
|
}
|
|
}
|
|
|
|
// The text that should be displayed for this counter.
|
|
void
|
|
nsCounterUseNode::GetText(nsString& aResult)
|
|
{
|
|
aResult.Truncate();
|
|
|
|
nsAutoVoidArray stack;
|
|
stack.AppendElement(static_cast<nsCounterNode*>(this));
|
|
|
|
if (mAllCounters && mScopeStart)
|
|
for (nsCounterNode *n = mScopeStart; n->mScopePrev; n = n->mScopeStart)
|
|
stack.AppendElement(n->mScopePrev);
|
|
|
|
PRInt32 style = mCounterStyle->Item(mAllCounters ? 2 : 1).GetIntValue();
|
|
const PRUnichar* separator;
|
|
if (mAllCounters)
|
|
separator = mCounterStyle->Item(1).GetStringBufferValue();
|
|
|
|
for (PRInt32 i = stack.Count() - 1;; --i) {
|
|
nsCounterNode *n = static_cast<nsCounterNode*>(stack[i]);
|
|
nsBulletFrame::AppendCounterText(style, n->mValueAfter, aResult);
|
|
if (i == 0)
|
|
break;
|
|
NS_ASSERTION(mAllCounters, "yikes, separator is uninitialized");
|
|
aResult.Append(separator);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCounterList::SetScope(nsCounterNode *aNode)
|
|
{
|
|
// This function is responsible for setting |mScopeStart| and
|
|
// |mScopePrev| (whose purpose is described in nsCounterManager.h).
|
|
// We do this by starting from the node immediately preceding
|
|
// |aNode| in content tree order, which is reasonably likely to be
|
|
// the previous element in our scope (or, for a reset, the previous
|
|
// element in the containing scope, which is what we want). If
|
|
// we're not in the same scope that it is, then it's too deep in the
|
|
// frame tree, so we walk up parent scopes until we find something
|
|
// appropriate.
|
|
|
|
if (aNode == First()) {
|
|
aNode->mScopeStart = nsnull;
|
|
aNode->mScopePrev = nsnull;
|
|
return;
|
|
}
|
|
|
|
// Get the content node for aNode's rendering object's *parent*,
|
|
// since scope includes siblings, so we want a descendant check on
|
|
// parents. If aNode is for a pseudo-element, then the parent
|
|
// rendering object is the frame's content; if aNode is for an
|
|
// element, then the parent rendering object is the frame's
|
|
// content's parent.
|
|
nsIContent *nodeContent = aNode->mPseudoFrame->GetContent();
|
|
if (!aNode->mPseudoFrame->GetStyleContext()->GetPseudoType()) {
|
|
nodeContent = nodeContent->GetParent();
|
|
}
|
|
|
|
for (nsCounterNode *prev = Prev(aNode), *start;
|
|
prev; prev = start->mScopePrev) {
|
|
// If |prev| starts a scope (because it's a real or implied
|
|
// reset), we want it as the scope start rather than the start
|
|
// of its enclosing scope. Otherwise, there's no enclosing
|
|
// scope, so the next thing in prev's scope shares its scope
|
|
// start.
|
|
start = (prev->mType == nsCounterNode::RESET || !prev->mScopeStart)
|
|
? prev : prev->mScopeStart;
|
|
|
|
// |startContent| is analogous to |nodeContent| (see above).
|
|
nsIContent *startContent = start->mPseudoFrame->GetContent();
|
|
if (!start->mPseudoFrame->GetStyleContext()->GetPseudoType()) {
|
|
startContent = startContent->GetParent();
|
|
}
|
|
NS_ASSERTION(nodeContent || !startContent,
|
|
"null check on startContent should be sufficient to "
|
|
"null check nodeContent as well, since if nodeContent "
|
|
"is for the root, startContent (which is before it) "
|
|
"must be too");
|
|
|
|
// A reset's outer scope can't be a scope created by a sibling.
|
|
if (!(aNode->mType == nsCounterNode::RESET &&
|
|
nodeContent == startContent) &&
|
|
// everything is inside the root (except the case above,
|
|
// a second reset on the root)
|
|
(!startContent ||
|
|
nsContentUtils::ContentIsDescendantOf(nodeContent,
|
|
startContent))) {
|
|
aNode->mScopeStart = start;
|
|
aNode->mScopePrev = prev;
|
|
return;
|
|
}
|
|
}
|
|
|
|
aNode->mScopeStart = nsnull;
|
|
aNode->mScopePrev = nsnull;
|
|
}
|
|
|
|
void
|
|
nsCounterList::RecalcAll()
|
|
{
|
|
mDirty = PR_FALSE;
|
|
|
|
nsCounterNode *node = First();
|
|
if (!node)
|
|
return;
|
|
|
|
do {
|
|
SetScope(node);
|
|
node->Calc(this);
|
|
|
|
if (node->mType == nsCounterNode::USE) {
|
|
nsCounterUseNode *useNode = node->UseNode();
|
|
// Null-check mText, since if the frame constructor isn't
|
|
// batching, we could end up here while the node is being
|
|
// constructed.
|
|
if (useNode->mText) {
|
|
nsAutoString text;
|
|
useNode->GetText(text);
|
|
useNode->mText->SetData(text);
|
|
}
|
|
}
|
|
} while ((node = Next(node)) != First());
|
|
}
|
|
|
|
nsCounterManager::nsCounterManager()
|
|
{
|
|
mNames.Init(16);
|
|
}
|
|
|
|
PRBool
|
|
nsCounterManager::AddCounterResetsAndIncrements(nsIFrame *aFrame)
|
|
{
|
|
const nsStyleContent *styleContent = aFrame->GetStyleContent();
|
|
if (!styleContent->CounterIncrementCount() &&
|
|
!styleContent->CounterResetCount())
|
|
return PR_FALSE;
|
|
|
|
// Add in order, resets first, so all the comparisons will be optimized
|
|
// for addition at the end of the list.
|
|
PRInt32 i, i_end;
|
|
PRBool dirty = PR_FALSE;
|
|
for (i = 0, i_end = styleContent->CounterResetCount(); i != i_end; ++i)
|
|
dirty |= AddResetOrIncrement(aFrame, i,
|
|
styleContent->GetCounterResetAt(i),
|
|
nsCounterChangeNode::RESET);
|
|
for (i = 0, i_end = styleContent->CounterIncrementCount(); i != i_end; ++i)
|
|
dirty |= AddResetOrIncrement(aFrame, i,
|
|
styleContent->GetCounterIncrementAt(i),
|
|
nsCounterChangeNode::INCREMENT);
|
|
return dirty;
|
|
}
|
|
|
|
PRBool
|
|
nsCounterManager::AddResetOrIncrement(nsIFrame *aFrame, PRInt32 aIndex,
|
|
const nsStyleCounterData *aCounterData,
|
|
nsCounterNode::Type aType)
|
|
{
|
|
nsCounterChangeNode *node =
|
|
new nsCounterChangeNode(aFrame, aType, aCounterData->mValue, aIndex);
|
|
if (!node)
|
|
return PR_FALSE;
|
|
|
|
nsCounterList *counterList = CounterListFor(aCounterData->mCounter);
|
|
if (!counterList) {
|
|
NS_NOTREACHED("CounterListFor failed (should only happen on OOM)");
|
|
return PR_FALSE;
|
|
}
|
|
|
|
counterList->Insert(node);
|
|
if (!counterList->IsLast(node)) {
|
|
// Tell the caller it's responsible for recalculating the entire
|
|
// list.
|
|
counterList->SetDirty();
|
|
return PR_TRUE;
|
|
}
|
|
|
|
// Don't call Calc() if the list is already dirty -- it'll be recalculated
|
|
// anyway, and trying to calculate with a dirty list doesn't work.
|
|
if (NS_LIKELY(!counterList->IsDirty())) {
|
|
node->Calc(counterList);
|
|
}
|
|
return PR_FALSE;
|
|
}
|
|
|
|
nsCounterList*
|
|
nsCounterManager::CounterListFor(const nsSubstring& aCounterName)
|
|
{
|
|
// XXX Why doesn't nsTHashtable provide an API that allows us to use
|
|
// get/put in one hashtable lookup?
|
|
nsCounterList *counterList;
|
|
if (!mNames.Get(aCounterName, &counterList)) {
|
|
counterList = new nsCounterList();
|
|
if (!counterList)
|
|
return nsnull;
|
|
if (!mNames.Put(aCounterName, counterList)) {
|
|
delete counterList;
|
|
return nsnull;
|
|
}
|
|
}
|
|
return counterList;
|
|
}
|
|
|
|
PR_STATIC_CALLBACK(PLDHashOperator)
|
|
RecalcDirtyLists(const nsAString& aKey, nsCounterList* aList, void* aClosure)
|
|
{
|
|
if (aList->IsDirty())
|
|
aList->RecalcAll();
|
|
return PL_DHASH_NEXT;
|
|
}
|
|
|
|
void
|
|
nsCounterManager::RecalcAll()
|
|
{
|
|
mNames.EnumerateRead(RecalcDirtyLists, nsnull);
|
|
}
|
|
|
|
struct DestroyNodesData {
|
|
DestroyNodesData(nsIFrame *aFrame)
|
|
: mFrame(aFrame)
|
|
, mDestroyedAny(PR_FALSE)
|
|
{
|
|
}
|
|
|
|
nsIFrame *mFrame;
|
|
PRBool mDestroyedAny;
|
|
};
|
|
|
|
PR_STATIC_CALLBACK(PLDHashOperator)
|
|
DestroyNodesInList(const nsAString& aKey, nsCounterList* aList, void* aClosure)
|
|
{
|
|
DestroyNodesData *data = static_cast<DestroyNodesData*>(aClosure);
|
|
if (aList->DestroyNodesFor(data->mFrame)) {
|
|
data->mDestroyedAny = PR_TRUE;
|
|
aList->SetDirty();
|
|
}
|
|
return PL_DHASH_NEXT;
|
|
}
|
|
|
|
PRBool
|
|
nsCounterManager::DestroyNodesFor(nsIFrame *aFrame)
|
|
{
|
|
DestroyNodesData data(aFrame);
|
|
mNames.EnumerateRead(DestroyNodesInList, &data);
|
|
return data.mDestroyedAny;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
PR_STATIC_CALLBACK(PLDHashOperator)
|
|
DumpList(const nsAString& aKey, nsCounterList* aList, void* aClosure)
|
|
{
|
|
printf("Counter named \"%s\":\n", NS_ConvertUTF16toUTF8(aKey).get());
|
|
nsCounterNode *node = aList->First();
|
|
|
|
if (node) {
|
|
PRInt32 i = 0;
|
|
do {
|
|
const char *types[] = { "RESET", "INCREMENT", "USE" };
|
|
printf(" Node #%d @%p frame=%p index=%d type=%s valAfter=%d\n"
|
|
" scope-start=%p scope-prev=%p",
|
|
i++, (void*)node, (void*)node->mPseudoFrame,
|
|
node->mContentIndex, types[node->mType], node->mValueAfter,
|
|
(void*)node->mScopeStart, (void*)node->mScopePrev);
|
|
if (node->mType == nsCounterNode::USE) {
|
|
nsAutoString text;
|
|
node->UseNode()->GetText(text);
|
|
printf(" text=%s", NS_ConvertUTF16toUTF8(text).get());
|
|
}
|
|
printf("\n");
|
|
} while ((node = aList->Next(node)) != aList->First());
|
|
}
|
|
return PL_DHASH_NEXT;
|
|
}
|
|
|
|
void
|
|
nsCounterManager::Dump()
|
|
{
|
|
printf("\n\nCounter Manager Lists:\n");
|
|
mNames.EnumerateRead(DumpList, nsnull);
|
|
printf("\n\n");
|
|
}
|
|
#endif
|