mirror of https://github.com/procxx/kepka.git
Fix selected messages copy with grouping.
This commit is contained in:
parent
4734700ac5
commit
963e969d2a
|
@ -87,6 +87,43 @@ int BinarySearchBlocksOrItems(const T &list, int edge) {
|
|||
|
||||
// flick scroll taken from http://qt-project.org/doc/qt-4.8/demos-embedded-anomaly-src-flickcharm-cpp.html
|
||||
|
||||
class HistoryInner::BotAbout : public ClickHandlerHost {
|
||||
public:
|
||||
BotAbout(not_null<HistoryInner*> parent, not_null<BotInfo*> info);
|
||||
|
||||
// ClickHandlerHost interface
|
||||
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
||||
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||
|
||||
not_null<BotInfo*> info;
|
||||
int width = 0;
|
||||
int height = 0;
|
||||
QRect rect;
|
||||
|
||||
private:
|
||||
not_null<HistoryInner*> _parent;
|
||||
|
||||
};
|
||||
|
||||
HistoryInner::BotAbout::BotAbout(
|
||||
not_null<HistoryInner*> parent,
|
||||
not_null<BotInfo*> info)
|
||||
: info(info)
|
||||
, _parent(parent) {
|
||||
}
|
||||
|
||||
void HistoryInner::BotAbout::clickHandlerActiveChanged(
|
||||
const ClickHandlerPtr &p,
|
||||
bool active) {
|
||||
_parent->update(rect);
|
||||
}
|
||||
|
||||
void HistoryInner::BotAbout::clickHandlerPressedChanged(
|
||||
const ClickHandlerPtr &p,
|
||||
bool pressed) {
|
||||
_parent->update(rect);
|
||||
}
|
||||
|
||||
HistoryInner::HistoryInner(
|
||||
not_null<HistoryWidget*> historyWidget,
|
||||
not_null<Window::Controller*> controller,
|
||||
|
@ -365,6 +402,44 @@ void HistoryInner::enumerateDates(Method method) {
|
|||
enumerateItems<EnumItemsDirection::BottomToTop>(dateCallback);
|
||||
}
|
||||
|
||||
TextSelection HistoryInner::computeRenderSelection(
|
||||
not_null<const SelectedItems*> selected,
|
||||
not_null<HistoryItem*> item) const {
|
||||
const auto itemSelection = [&](not_null<HistoryItem*> item) {
|
||||
auto i = selected->find(item);
|
||||
if (i != selected->end()) {
|
||||
return i->second;
|
||||
}
|
||||
return TextSelection();
|
||||
};
|
||||
const auto group = item->Get<HistoryMessageGroup>();
|
||||
if (group) {
|
||||
if (group->leader != item) {
|
||||
return TextSelection();
|
||||
}
|
||||
auto result = TextSelection();
|
||||
auto allFullSelected = true;
|
||||
const auto count = int(group->others.size());
|
||||
for (auto i = 0; i != count; ++i) {
|
||||
if (itemSelection(group->others[i]) == FullSelection) {
|
||||
result = AddGroupItemSelection(result, i);
|
||||
} else {
|
||||
allFullSelected = false;
|
||||
}
|
||||
}
|
||||
const auto leaderSelection = itemSelection(item);
|
||||
if (leaderSelection == FullSelection) {
|
||||
return allFullSelected
|
||||
? FullSelection
|
||||
: AddGroupItemSelection(result, count);
|
||||
} else if (leaderSelection != TextSelection()) {
|
||||
return leaderSelection;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
return itemSelection(item);
|
||||
}
|
||||
|
||||
TextSelection HistoryInner::itemRenderSelection(
|
||||
not_null<HistoryItem*> item,
|
||||
int selfromy,
|
||||
|
@ -377,39 +452,7 @@ TextSelection HistoryInner::itemRenderSelection(
|
|||
return FullSelection;
|
||||
}
|
||||
} else if (!_selected.empty()) {
|
||||
const auto itemSelection = [&](not_null<HistoryItem*> item) {
|
||||
auto i = _selected.find(item);
|
||||
if (i != _selected.end()) {
|
||||
return i->second;
|
||||
}
|
||||
return TextSelection();
|
||||
};
|
||||
const auto group = item->Get<HistoryMessageGroup>();
|
||||
if (group) {
|
||||
if (group->leader != item) {
|
||||
return TextSelection();
|
||||
}
|
||||
auto result = TextSelection();
|
||||
auto allFullSelected = true;
|
||||
const auto count = int(group->others.size());
|
||||
for (auto i = 0; i != count; ++i) {
|
||||
if (itemSelection(group->others[i]) == FullSelection) {
|
||||
result = AddGroupItemSelection(result, i);
|
||||
} else {
|
||||
allFullSelected = false;
|
||||
}
|
||||
}
|
||||
const auto leaderSelection = itemSelection(item);
|
||||
if (leaderSelection == FullSelection) {
|
||||
return allFullSelected
|
||||
? FullSelection
|
||||
: AddGroupItemSelection(result, count);
|
||||
} else if (leaderSelection != TextSelection()) {
|
||||
return leaderSelection;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
return itemSelection(item);
|
||||
return computeRenderSelection(&_selected, item);
|
||||
}
|
||||
return TextSelection();
|
||||
}
|
||||
|
@ -1605,8 +1648,9 @@ void HistoryInner::copyContextText() {
|
|||
if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) {
|
||||
return;
|
||||
}
|
||||
|
||||
setToClipboard(item->selectedText(FullSelection));
|
||||
const auto group = item->getFullGroup();
|
||||
const auto leader = group ? group->leader : item;
|
||||
setToClipboard(leader->selectedText(FullSelection));
|
||||
}
|
||||
|
||||
void HistoryInner::setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode) {
|
||||
|
@ -1634,32 +1678,65 @@ TextWithEntities HistoryInner::getSelectedText() const {
|
|||
return item->selectedText(selection);
|
||||
}
|
||||
|
||||
int fullSize = 0;
|
||||
QString timeFormat(qsl(", [dd.MM.yy hh:mm]\n"));
|
||||
QMap<int, TextWithEntities> texts;
|
||||
for (const auto [item, selection] : selected) {
|
||||
if (item->detached()) continue;
|
||||
const auto timeFormat = qsl(", [dd.MM.yy hh:mm]\n");
|
||||
auto groupLeadersAdded = base::flat_set<not_null<HistoryItem*>>();
|
||||
auto fullSize = 0;
|
||||
auto texts = base::flat_map<std::pair<int, MsgId>, TextWithEntities>();
|
||||
|
||||
const auto addItem = [&](
|
||||
not_null<HistoryItem*> item,
|
||||
TextSelection selection) {
|
||||
auto time = item->date.toString(timeFormat);
|
||||
TextWithEntities part, unwrapped = item->selectedText(FullSelection);
|
||||
int size = item->author()->name.size() + time.size() + unwrapped.text.size();
|
||||
auto part = TextWithEntities();
|
||||
auto unwrapped = item->selectedText(selection);
|
||||
auto size = item->author()->name.size()
|
||||
+ time.size()
|
||||
+ unwrapped.text.size();
|
||||
part.text.reserve(size);
|
||||
|
||||
int y = itemTop(item);
|
||||
auto y = itemTop(item);
|
||||
if (y >= 0) {
|
||||
part.text.append(item->author()->name).append(time);
|
||||
TextUtilities::Append(part, std::move(unwrapped));
|
||||
texts.insert(y, part);
|
||||
texts.emplace(std::make_pair(y, item->id), part);
|
||||
fullSize += size;
|
||||
}
|
||||
};
|
||||
|
||||
for (const auto [item, selection] : selected) {
|
||||
if (item->detached()) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (const auto group = item->Get<HistoryMessageGroup>()) {
|
||||
if (groupLeadersAdded.contains(group->leader)) {
|
||||
continue;
|
||||
}
|
||||
const auto leaderSelection = computeRenderSelection(
|
||||
&selected,
|
||||
group->leader);
|
||||
if (leaderSelection == FullSelection) {
|
||||
groupLeadersAdded.emplace(group->leader);
|
||||
addItem(group->leader, FullSelection);
|
||||
} else if (item == group->leader) {
|
||||
const auto leaderFullSelection = AddGroupItemSelection(
|
||||
TextSelection(),
|
||||
int(group->others.size()));
|
||||
addItem(item, leaderFullSelection);
|
||||
} else {
|
||||
addItem(item, FullSelection);
|
||||
}
|
||||
} else {
|
||||
addItem(item, FullSelection);
|
||||
}
|
||||
}
|
||||
|
||||
TextWithEntities result;
|
||||
auto result = TextWithEntities();
|
||||
auto sep = qsl("\n\n");
|
||||
result.text.reserve(fullSize + (texts.size() - 1) * sep.size());
|
||||
for (auto i = texts.begin(), e = texts.end(); i != e; ++i) {
|
||||
TextUtilities::Append(result, std::move(i.value()));
|
||||
if (i + 1 != e) {
|
||||
for (auto i = texts.begin(), e = texts.end(); i != e;) {
|
||||
TextUtilities::Append(result, std::move(i->second));
|
||||
if (++i != e) {
|
||||
result.text.append(sep);
|
||||
}
|
||||
}
|
||||
|
@ -2386,14 +2463,6 @@ void HistoryInner::updateDragSelection(HistoryItem *dragSelFrom, HistoryItem *dr
|
|||
update();
|
||||
}
|
||||
|
||||
void HistoryInner::BotAbout::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) {
|
||||
_parent->update(rect);
|
||||
}
|
||||
|
||||
void HistoryInner::BotAbout::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
|
||||
_parent->update(rect);
|
||||
}
|
||||
|
||||
int HistoryInner::historyHeight() const {
|
||||
int result = 0;
|
||||
if (!_history || _history->isEmpty()) {
|
||||
|
@ -2441,13 +2510,16 @@ int HistoryInner::itemTop(const HistoryItem *item) const { // -1 if should not b
|
|||
}
|
||||
|
||||
void HistoryInner::notifyIsBotChanged() {
|
||||
BotInfo *newinfo = (_history && _history->peer->isUser()) ? _history->peer->asUser()->botInfo.get() : nullptr;
|
||||
if ((!newinfo && !_botAbout) || (newinfo && _botAbout && _botAbout->info == newinfo)) {
|
||||
const auto newinfo = (_history && _history->peer->isUser())
|
||||
? _history->peer->asUser()->botInfo.get()
|
||||
: nullptr;
|
||||
if ((!newinfo && !_botAbout)
|
||||
|| (newinfo && _botAbout && _botAbout->info == newinfo)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (newinfo) {
|
||||
_botAbout.reset(new BotAbout(this, newinfo));
|
||||
_botAbout = std::make_unique<BotAbout>(this, newinfo);
|
||||
if (newinfo && !newinfo->inited) {
|
||||
Auth().api().requestFullPeer(_peer);
|
||||
}
|
||||
|
|
|
@ -133,6 +133,9 @@ private slots:
|
|||
void onScrollDateHideByTimer();
|
||||
|
||||
private:
|
||||
class BotAbout;
|
||||
using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>;
|
||||
|
||||
enum class MouseAction {
|
||||
None,
|
||||
PrepareDrag,
|
||||
|
@ -140,7 +143,6 @@ private:
|
|||
PrepareSelect,
|
||||
Selecting,
|
||||
};
|
||||
|
||||
enum class SelectAction {
|
||||
Select,
|
||||
Deselect,
|
||||
|
@ -175,6 +177,9 @@ private:
|
|||
not_null<HistoryItem*> item,
|
||||
int selfromy,
|
||||
int seltoy) const;
|
||||
TextSelection computeRenderSelection(
|
||||
not_null<const SelectedItems*> selected,
|
||||
not_null<HistoryItem*> item) const;
|
||||
|
||||
void setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode = QClipboard::Clipboard);
|
||||
|
||||
|
@ -196,23 +201,6 @@ private:
|
|||
// or at least we don't need to display first _history date (just skip it by height)
|
||||
int _historySkipHeight = 0;
|
||||
|
||||
class BotAbout : public ClickHandlerHost {
|
||||
public:
|
||||
BotAbout(HistoryInner *parent, BotInfo *info) : info(info), _parent(parent) {
|
||||
}
|
||||
BotInfo *info = nullptr;
|
||||
int width = 0;
|
||||
int height = 0;
|
||||
QRect rect;
|
||||
|
||||
// ClickHandlerHost interface
|
||||
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
||||
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||
|
||||
private:
|
||||
HistoryInner *_parent;
|
||||
|
||||
};
|
||||
std::unique_ptr<BotAbout> _botAbout;
|
||||
|
||||
HistoryWidget *_widget = nullptr;
|
||||
|
@ -224,7 +212,6 @@ private:
|
|||
bool _firstLoading = false;
|
||||
|
||||
style::cursor _cursor = style::cur_default;
|
||||
using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>;
|
||||
SelectedItems _selected;
|
||||
|
||||
void applyDragSelection();
|
||||
|
|
|
@ -288,10 +288,15 @@ QString HistoryGroupedMedia::inDialogsText() const {
|
|||
|
||||
TextWithEntities HistoryGroupedMedia::selectedText(
|
||||
TextSelection selection) const {
|
||||
return WithCaptionSelectedText(
|
||||
lang(lng_in_dlg_album),
|
||||
_caption,
|
||||
selection);
|
||||
if (!IsSubGroupSelection(selection)) {
|
||||
return WithCaptionSelectedText(
|
||||
lang(lng_in_dlg_album),
|
||||
_caption,
|
||||
selection);
|
||||
} else if (IsGroupItemSelection(selection, int(_elements.size()) - 1)) {
|
||||
return main()->selectedText(FullSelection);
|
||||
}
|
||||
return TextWithEntities();
|
||||
}
|
||||
|
||||
void HistoryGroupedMedia::clickHandlerActiveChanged(
|
||||
|
|
|
@ -1539,7 +1539,9 @@ TextWithEntities HistoryMessage::selectedText(TextSelection selection) const {
|
|||
ExpandLinksAll);
|
||||
auto skipped = skipTextSelection(selection);
|
||||
auto mediaDisplayed = (_media && _media->isDisplayed());
|
||||
auto mediaResult = mediaDisplayed ? _media->selectedText(skipped) : TextWithEntities();
|
||||
auto mediaResult = (mediaDisplayed || isHiddenByGroup())
|
||||
? _media->selectedText(skipped)
|
||||
: TextWithEntities();
|
||||
if (auto entry = Get<HistoryMessageLogEntryOriginal>()) {
|
||||
const auto originalSelection = mediaDisplayed
|
||||
? _media->skipSelection(skipped)
|
||||
|
|
Loading…
Reference in New Issue