/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim:set ts=2 sw=2 sts=2 et cindent: */ /* 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 "DocumentL10n.h" #include "nsIContentSink.h" #include "mozilla/dom/DocumentL10nBinding.h" using namespace mozilla::dom; NS_IMPL_CYCLE_COLLECTION_CLASS(DocumentL10n) NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(DocumentL10n, DOMLocalization) NS_IMPL_CYCLE_COLLECTION_UNLINK(mDocument) NS_IMPL_CYCLE_COLLECTION_UNLINK(mReady) NS_IMPL_CYCLE_COLLECTION_UNLINK(mContentSink) NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(DocumentL10n, DOMLocalization) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDocument) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mReady) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mContentSink) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_IMPL_ADDREF_INHERITED(DocumentL10n, DOMLocalization) NS_IMPL_RELEASE_INHERITED(DocumentL10n, DOMLocalization) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(DocumentL10n) NS_INTERFACE_MAP_END_INHERITING(DOMLocalization) DocumentL10n::DocumentL10n(Document* aDocument) : DOMLocalization(aDocument->GetScopeObject()), mDocument(aDocument), mState(DocumentL10nState::Initialized) { mContentSink = do_QueryInterface(aDocument->GetCurrentContentSink()); Element* elem = mDocument->GetDocumentElement(); if (elem) { mIsSync = elem->HasAttr(kNameSpaceID_None, nsGkAtoms::datal10nsync); } } void DocumentL10n::Init(nsTArray& aResourceIds, ErrorResult& aRv) { DOMLocalization::Init(aResourceIds, aRv); if (NS_WARN_IF(aRv.Failed())) { return; } mReady = Promise::Create(mGlobal, aRv); if (NS_WARN_IF(aRv.Failed())) { return; } } JSObject* DocumentL10n::WrapObject(JSContext* aCx, JS::Handle aGivenProto) { return DocumentL10n_Binding::Wrap(aCx, this, aGivenProto); } class L10nReadyHandler final : public PromiseNativeHandler { public: NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_CLASS(L10nReadyHandler) explicit L10nReadyHandler(Promise* aPromise, DocumentL10n* aDocumentL10n) : mPromise(aPromise), mDocumentL10n(aDocumentL10n) {} void ResolvedCallback(JSContext* aCx, JS::Handle aValue) override { mDocumentL10n->InitialDocumentTranslationCompleted(); mPromise->MaybeResolveWithUndefined(); } void RejectedCallback(JSContext* aCx, JS::Handle aValue) override { mDocumentL10n->InitialDocumentTranslationCompleted(); mPromise->MaybeRejectWithUndefined(); } private: ~L10nReadyHandler() = default; RefPtr mPromise; RefPtr mDocumentL10n; }; NS_IMPL_CYCLE_COLLECTION(L10nReadyHandler, mPromise, mDocumentL10n) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(L10nReadyHandler) NS_INTERFACE_MAP_ENTRY(nsISupports) NS_INTERFACE_MAP_END NS_IMPL_CYCLE_COLLECTING_ADDREF(L10nReadyHandler) NS_IMPL_CYCLE_COLLECTING_RELEASE(L10nReadyHandler) void DocumentL10n::TriggerInitialDocumentTranslation() { if (mState >= DocumentL10nState::InitialTranslationTriggered) { return; } mState = DocumentL10nState::InitialTranslationTriggered; Element* elem = mDocument->GetDocumentElement(); if (!elem) { InitialDocumentTranslationCompleted(); mReady->MaybeRejectWithUndefined(); return; } ErrorResult rv; // 1. Collect all localizable elements. Sequence> elements; GetTranslatables(*elem, elements, rv); if (NS_WARN_IF(rv.Failed())) { InitialDocumentTranslationCompleted(); mReady->MaybeRejectWithUndefined(); return; } RefPtr proto = mDocument->GetPrototype(); RefPtr promise; // 2. Check if the document has a prototype that may cache // translated elements. if (proto) { // 2.1. Handle the case when we have proto. // 2.1.1. Move elements that are not in the proto to a separate // array. Sequence> nonProtoElements; uint32_t i = elements.Length(); while (i > 0) { Element* elem = elements.ElementAt(i - 1); MOZ_RELEASE_ASSERT(elem->HasAttr(nsGkAtoms::datal10nid)); if (!elem->HasElementCreatedFromPrototypeAndHasUnmodifiedL10n()) { nonProtoElements.AppendElement(*elem, fallible); elements.RemoveElement(elem); } i--; } // We populate the sequence in reverse order. Let's bring it // back to top->bottom one. nonProtoElements.Reverse(); nsTArray> promises; // 2.1.2. If we're not loading from cache, push the elements that // are in the prototype to be translated and cached. if (!proto->WasL10nCached() && !elements.IsEmpty()) { RefPtr translatePromise = TranslateElements(elements, proto, rv); if (NS_WARN_IF(!translatePromise || rv.Failed())) { InitialDocumentTranslationCompleted(); mReady->MaybeRejectWithUndefined(); return; } promises.AppendElement(translatePromise); } // 2.1.3. If there are elements that are not in the prototype, // localize them without attempting to cache and // independently of if we're loading from cache. if (!nonProtoElements.IsEmpty()) { RefPtr nonProtoTranslatePromise = TranslateElements(nonProtoElements, nullptr, rv); if (NS_WARN_IF(!nonProtoTranslatePromise || rv.Failed())) { InitialDocumentTranslationCompleted(); mReady->MaybeRejectWithUndefined(); return; } promises.AppendElement(nonProtoTranslatePromise); } // 2.1.4. Collect promises with Promise::All (maybe empty). AutoEntryScript aes(mGlobal, "DocumentL10n InitialDocumentTranslationCompleted"); promise = Promise::All(aes.cx(), promises, rv); } else { // 2.2. Handle the case when we don't have proto. // 2.2.1. Otherwise, translate all available elements, // without attempting to cache them. promise = TranslateElements(elements, nullptr, rv); } if (NS_WARN_IF(!promise || rv.Failed())) { InitialDocumentTranslationCompleted(); mReady->MaybeRejectWithUndefined(); return; } // 3. Connect the root to L10nMutations observer. ConnectRoot(*elem, rv); // 4. Check if the promise is already resolved. if (promise->State() == Promise::PromiseState::Resolved) { // 4.1. If it is, resolved immediatelly. InitialDocumentTranslationCompleted(); mReady->MaybeResolveWithUndefined(); } else { // 4.2. If not, schedule the L10nReadyHandler. RefPtr l10nReadyHandler = new L10nReadyHandler(mReady, this); promise->AppendNativeHandler(l10nReadyHandler); } } void DocumentL10n::InitialDocumentTranslationCompleted() { if (mState >= DocumentL10nState::InitialTranslationCompleted) { return; } Element* documentElement = mDocument->GetDocumentElement(); if (documentElement) { SetRootInfo(documentElement); } mState = DocumentL10nState::InitialTranslationCompleted; mDocument->InitialDocumentTranslationCompleted(); // In XUL scenario contentSink is nullptr. if (mContentSink) { mContentSink->InitialDocumentTranslationCompleted(); } // If sync was true, we want to change the state of // mozILocalization to async now. if (mIsSync) { mIsSync = false; mLocalization->SetIsSync(mIsSync); } } Promise* DocumentL10n::Ready() { return mReady; } void DocumentL10n::OnCreatePresShell() { mMutations->OnCreatePresShell(); }