1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
|
// 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.
#include "ui/views/controls/link_fragment.h"
#include <string>
#include "ui/base/metadata/metadata_impl_macros.h"
#include "ui/views/controls/focus_ring.h"
#include "ui/views/controls/link.h"
#include "ui/views/style/typography.h"
#include "ui/views/view_utils.h"
namespace views {
LinkFragment::LinkFragment(const std::u16string& title,
int text_context,
int text_style,
LinkFragment* other_fragment)
: Link(title, text_context, text_style),
prev_fragment_(this),
next_fragment_(this) {
// Connect to the previous fragment if it exists.
if (other_fragment) {
Connect(other_fragment);
}
views::FocusRing::Install(this);
views::FocusRing::Get(this)->SetHasFocusPredicate(
base::BindRepeating([](const View* view) {
const auto* v = views::AsViewClass<LinkFragment>(view);
CHECK(v);
return InvokeOnFragments(&LinkFragment::HasFocus, v);
}));
}
LinkFragment::~LinkFragment() {
Disconnect();
}
void LinkFragment::Connect(LinkFragment* other_fragment) {
DCHECK(prev_fragment_ == this);
DCHECK(next_fragment_ == this);
DCHECK(other_fragment);
next_fragment_ = other_fragment->next_fragment_;
other_fragment->next_fragment_->prev_fragment_ = this;
prev_fragment_ = other_fragment;
other_fragment->next_fragment_ = this;
}
void LinkFragment::Disconnect() {
DCHECK((prev_fragment_ != this) == (next_fragment_ != this));
if (prev_fragment_ != this) {
prev_fragment_->next_fragment_ = next_fragment_;
next_fragment_->prev_fragment_ = prev_fragment_;
}
}
bool LinkFragment::IsUnderlined() const {
return GetEnabled() &&
(HasFocus() || IsMouseHovered() || GetForceUnderline());
}
void LinkFragment::RecalculateFont() {
// Check whether any link fragment should be underlined.
const bool should_be_underlined =
InvokeOnFragments(&LinkFragment::IsUnderlined, this);
InvokeOnFragments(
[should_be_underlined](LinkFragment* fragment) {
// If the style differs from the current one, update.
if (!!(fragment->font_list().GetFontStyle() & gfx::Font::UNDERLINE) !=
should_be_underlined) {
const int style = fragment->font_list().GetFontStyle();
const int intended_style = should_be_underlined
? (style | gfx::Font::UNDERLINE)
: (style & ~gfx::Font::UNDERLINE);
fragment->Label::SetFontList(
fragment->font_list().DeriveWithStyle(intended_style));
fragment->SchedulePaint();
}
views::FocusRing::Get(fragment)->SchedulePaint();
return false;
},
this);
}
BEGIN_METADATA(LinkFragment)
END_METADATA
} // namespace views
|