chromium/ios/web/annotations/annotations_text_manager_impl.mm

// Copyright 2022 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#import "ios/web/annotations/annotations_text_manager_impl.h"

#import "base/strings/string_util.h"
#import "ios/web/annotations/annotations_java_script_feature.h"
#import "ios/web/common/features.h"
#import "ios/web/common/url_scheme_util.h"
#import "ios/web/public/annotations/annotations_text_observer.h"
#import "ios/web/public/js_messaging/web_frame.h"
#import "ios/web/public/navigation/navigation_context.h"
#import "ios/web/public/web_state.h"

namespace web {

static const int kMaxAnnotationsTextLength = 65535;

AnnotationsTextManagerImpl::AnnotationsTextManagerImpl(WebState* web_state)
    : web_state_(web_state),
      seq_id_(1),
      is_viewport_extraction_(
          base::FeatureList::IsEnabled(features::kEnableViewportIntents)) {
  DCHECK(web_state_);
  web_state_->AddObserver(this);
}

AnnotationsTextManagerImpl::~AnnotationsTextManagerImpl() {
  web_state_ = nullptr;
}

void AnnotationsTextManagerImpl::AddObserver(
    AnnotationsTextObserver* observer) {
  observers_.AddObserver(observer);
}

void AnnotationsTextManagerImpl::RemoveObserver(
    AnnotationsTextObserver* observer) {
  observers_.RemoveObserver(observer);
}

void AnnotationsTextManagerImpl::DecorateAnnotations(WebState* web_state,
                                                     base::Value& annotations,
                                                     int seq_id) {
  DCHECK_EQ(web_state_, web_state);
  // This can happen if `RemoveDecorations` is called before this is called.
  if (!is_viewport_extraction_ && seq_id != seq_id_) {
    return;
  }
  AnnotationsJavaScriptFeature::GetInstance()->DecorateAnnotations(
      web_state_, annotations, seq_id);
}

void AnnotationsTextManagerImpl::RemoveDecorations() {
  seq_id_++;
  AnnotationsJavaScriptFeature::GetInstance()->RemoveDecorations(web_state_);
}

void AnnotationsTextManagerImpl::RemoveDecorationsWithType(
    const std::string& type) {
  seq_id_++;
  AnnotationsJavaScriptFeature::GetInstance()->RemoveDecorationsWithType(
      web_state_, type);
}

void AnnotationsTextManagerImpl::RemoveHighlight() {
  AnnotationsJavaScriptFeature::GetInstance()->RemoveHighlight(web_state_);
}

void AnnotationsTextManagerImpl::StartExtractingText() {
  DCHECK(web_state_);
  const GURL& url = web_state_->GetVisibleURL();
  if (observers_.empty() || !web::UrlHasWebScheme(url) ||
      !web_state_->ContentIsHTML()) {
    return;
  }

  seq_id_++;
  AnnotationsJavaScriptFeature::GetInstance()->ExtractText(
      web_state_, kMaxAnnotationsTextLength, seq_id_);
}

void AnnotationsTextManagerImpl::SetSupportedTypes(
    NSTextCheckingType supported_types) {
  supported_types_ = supported_types;
}

#pragma mark - WebStateObserver methods.

void AnnotationsTextManagerImpl::PageLoaded(
    web::WebState* web_state,
    web::PageLoadCompletionStatus load_completion_status) {
  DCHECK_EQ(web_state_, web_state);
  if (load_completion_status == web::PageLoadCompletionStatus::SUCCESS &&
      supported_types_) {
    base::SequencedTaskRunner::GetCurrentDefault()->PostTask(
        FROM_HERE,
        base::BindOnce(&AnnotationsTextManagerImpl::StartExtractingText,
                       weak_factory_.GetWeakPtr()));
  }
}

void AnnotationsTextManagerImpl::WebStateDestroyed(WebState* web_state) {
  web_state_->RemoveObserver(this);
  web_state_ = nullptr;
}

#pragma mark - JS Methods

void AnnotationsTextManagerImpl::OnTextExtracted(
    WebState* web_state,
    const std::string& text,
    int seq_id,
    const base::Value::Dict& metadata) {
  if (!web_state_ || (!is_viewport_extraction_ && seq_id != seq_id_)) {
    return;
  }
  DCHECK(web_state_ == web_state);
  for (auto& observer : observers_) {
    observer.OnTextExtracted(web_state, text, seq_id, metadata);
  }
}

void AnnotationsTextManagerImpl::OnDecorated(
    WebState* web_state,
    int annotations,
    int successes,
    int failures,
    const base::Value::List& cancelled) {
  if (!web_state_) {
    return;
  }
  DCHECK(web_state_ == web_state);
  for (auto& observer : observers_) {
    observer.OnDecorated(web_state, annotations, successes, failures,
                         cancelled);
  }
}

void AnnotationsTextManagerImpl::OnClick(WebState* web_state,
                                         const std::string& text,
                                         CGRect rect,
                                         const std::string& data) {
  if (!web_state_) {
    return;
  }
  DCHECK(web_state_ == web_state);
  for (auto& observer : observers_) {
    observer.OnClick(web_state, text, rect, data);
  }
}

WEB_STATE_USER_DATA_KEY_IMPL(AnnotationsTextManager)

}  // namespace web