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
|
// 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(
|
HistoryInner::HistoryInner(
|
||||||
not_null<HistoryWidget*> historyWidget,
|
not_null<HistoryWidget*> historyWidget,
|
||||||
not_null<Window::Controller*> controller,
|
not_null<Window::Controller*> controller,
|
||||||
|
@ -365,6 +402,44 @@ void HistoryInner::enumerateDates(Method method) {
|
||||||
enumerateItems<EnumItemsDirection::BottomToTop>(dateCallback);
|
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(
|
TextSelection HistoryInner::itemRenderSelection(
|
||||||
not_null<HistoryItem*> item,
|
not_null<HistoryItem*> item,
|
||||||
int selfromy,
|
int selfromy,
|
||||||
|
@ -377,39 +452,7 @@ TextSelection HistoryInner::itemRenderSelection(
|
||||||
return FullSelection;
|
return FullSelection;
|
||||||
}
|
}
|
||||||
} else if (!_selected.empty()) {
|
} else if (!_selected.empty()) {
|
||||||
const auto itemSelection = [&](not_null<HistoryItem*> item) {
|
return computeRenderSelection(&_selected, 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 TextSelection();
|
return TextSelection();
|
||||||
}
|
}
|
||||||
|
@ -1605,8 +1648,9 @@ void HistoryInner::copyContextText() {
|
||||||
if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) {
|
if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const auto group = item->getFullGroup();
|
||||||
setToClipboard(item->selectedText(FullSelection));
|
const auto leader = group ? group->leader : item;
|
||||||
|
setToClipboard(leader->selectedText(FullSelection));
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryInner::setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode) {
|
void HistoryInner::setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode) {
|
||||||
|
@ -1634,32 +1678,65 @@ TextWithEntities HistoryInner::getSelectedText() const {
|
||||||
return item->selectedText(selection);
|
return item->selectedText(selection);
|
||||||
}
|
}
|
||||||
|
|
||||||
int fullSize = 0;
|
const auto timeFormat = qsl(", [dd.MM.yy hh:mm]\n");
|
||||||
QString timeFormat(qsl(", [dd.MM.yy hh:mm]\n"));
|
auto groupLeadersAdded = base::flat_set<not_null<HistoryItem*>>();
|
||||||
QMap<int, TextWithEntities> texts;
|
auto fullSize = 0;
|
||||||
for (const auto [item, selection] : selected) {
|
auto texts = base::flat_map<std::pair<int, MsgId>, TextWithEntities>();
|
||||||
if (item->detached()) continue;
|
|
||||||
|
|
||||||
|
const auto addItem = [&](
|
||||||
|
not_null<HistoryItem*> item,
|
||||||
|
TextSelection selection) {
|
||||||
auto time = item->date.toString(timeFormat);
|
auto time = item->date.toString(timeFormat);
|
||||||
TextWithEntities part, unwrapped = item->selectedText(FullSelection);
|
auto part = TextWithEntities();
|
||||||
int size = item->author()->name.size() + time.size() + unwrapped.text.size();
|
auto unwrapped = item->selectedText(selection);
|
||||||
|
auto size = item->author()->name.size()
|
||||||
|
+ time.size()
|
||||||
|
+ unwrapped.text.size();
|
||||||
part.text.reserve(size);
|
part.text.reserve(size);
|
||||||
|
|
||||||
int y = itemTop(item);
|
auto y = itemTop(item);
|
||||||
if (y >= 0) {
|
if (y >= 0) {
|
||||||
part.text.append(item->author()->name).append(time);
|
part.text.append(item->author()->name).append(time);
|
||||||
TextUtilities::Append(part, std::move(unwrapped));
|
TextUtilities::Append(part, std::move(unwrapped));
|
||||||
texts.insert(y, part);
|
texts.emplace(std::make_pair(y, item->id), part);
|
||||||
fullSize += size;
|
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");
|
auto sep = qsl("\n\n");
|
||||||
result.text.reserve(fullSize + (texts.size() - 1) * sep.size());
|
result.text.reserve(fullSize + (texts.size() - 1) * sep.size());
|
||||||
for (auto i = texts.begin(), e = texts.end(); i != e; ++i) {
|
for (auto i = texts.begin(), e = texts.end(); i != e;) {
|
||||||
TextUtilities::Append(result, std::move(i.value()));
|
TextUtilities::Append(result, std::move(i->second));
|
||||||
if (i + 1 != e) {
|
if (++i != e) {
|
||||||
result.text.append(sep);
|
result.text.append(sep);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2386,14 +2463,6 @@ void HistoryInner::updateDragSelection(HistoryItem *dragSelFrom, HistoryItem *dr
|
||||||
update();
|
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 HistoryInner::historyHeight() const {
|
||||||
int result = 0;
|
int result = 0;
|
||||||
if (!_history || _history->isEmpty()) {
|
if (!_history || _history->isEmpty()) {
|
||||||
|
@ -2441,13 +2510,16 @@ int HistoryInner::itemTop(const HistoryItem *item) const { // -1 if should not b
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryInner::notifyIsBotChanged() {
|
void HistoryInner::notifyIsBotChanged() {
|
||||||
BotInfo *newinfo = (_history && _history->peer->isUser()) ? _history->peer->asUser()->botInfo.get() : nullptr;
|
const auto newinfo = (_history && _history->peer->isUser())
|
||||||
if ((!newinfo && !_botAbout) || (newinfo && _botAbout && _botAbout->info == newinfo)) {
|
? _history->peer->asUser()->botInfo.get()
|
||||||
|
: nullptr;
|
||||||
|
if ((!newinfo && !_botAbout)
|
||||||
|
|| (newinfo && _botAbout && _botAbout->info == newinfo)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newinfo) {
|
if (newinfo) {
|
||||||
_botAbout.reset(new BotAbout(this, newinfo));
|
_botAbout = std::make_unique<BotAbout>(this, newinfo);
|
||||||
if (newinfo && !newinfo->inited) {
|
if (newinfo && !newinfo->inited) {
|
||||||
Auth().api().requestFullPeer(_peer);
|
Auth().api().requestFullPeer(_peer);
|
||||||
}
|
}
|
||||||
|
|
|
@ -133,6 +133,9 @@ private slots:
|
||||||
void onScrollDateHideByTimer();
|
void onScrollDateHideByTimer();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
class BotAbout;
|
||||||
|
using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>;
|
||||||
|
|
||||||
enum class MouseAction {
|
enum class MouseAction {
|
||||||
None,
|
None,
|
||||||
PrepareDrag,
|
PrepareDrag,
|
||||||
|
@ -140,7 +143,6 @@ private:
|
||||||
PrepareSelect,
|
PrepareSelect,
|
||||||
Selecting,
|
Selecting,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class SelectAction {
|
enum class SelectAction {
|
||||||
Select,
|
Select,
|
||||||
Deselect,
|
Deselect,
|
||||||
|
@ -175,6 +177,9 @@ private:
|
||||||
not_null<HistoryItem*> item,
|
not_null<HistoryItem*> item,
|
||||||
int selfromy,
|
int selfromy,
|
||||||
int seltoy) const;
|
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);
|
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)
|
// or at least we don't need to display first _history date (just skip it by height)
|
||||||
int _historySkipHeight = 0;
|
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;
|
std::unique_ptr<BotAbout> _botAbout;
|
||||||
|
|
||||||
HistoryWidget *_widget = nullptr;
|
HistoryWidget *_widget = nullptr;
|
||||||
|
@ -224,7 +212,6 @@ private:
|
||||||
bool _firstLoading = false;
|
bool _firstLoading = false;
|
||||||
|
|
||||||
style::cursor _cursor = style::cur_default;
|
style::cursor _cursor = style::cur_default;
|
||||||
using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>;
|
|
||||||
SelectedItems _selected;
|
SelectedItems _selected;
|
||||||
|
|
||||||
void applyDragSelection();
|
void applyDragSelection();
|
||||||
|
|
|
@ -288,10 +288,15 @@ QString HistoryGroupedMedia::inDialogsText() const {
|
||||||
|
|
||||||
TextWithEntities HistoryGroupedMedia::selectedText(
|
TextWithEntities HistoryGroupedMedia::selectedText(
|
||||||
TextSelection selection) const {
|
TextSelection selection) const {
|
||||||
return WithCaptionSelectedText(
|
if (!IsSubGroupSelection(selection)) {
|
||||||
lang(lng_in_dlg_album),
|
return WithCaptionSelectedText(
|
||||||
_caption,
|
lang(lng_in_dlg_album),
|
||||||
selection);
|
_caption,
|
||||||
|
selection);
|
||||||
|
} else if (IsGroupItemSelection(selection, int(_elements.size()) - 1)) {
|
||||||
|
return main()->selectedText(FullSelection);
|
||||||
|
}
|
||||||
|
return TextWithEntities();
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryGroupedMedia::clickHandlerActiveChanged(
|
void HistoryGroupedMedia::clickHandlerActiveChanged(
|
||||||
|
|
|
@ -1539,7 +1539,9 @@ TextWithEntities HistoryMessage::selectedText(TextSelection selection) const {
|
||||||
ExpandLinksAll);
|
ExpandLinksAll);
|
||||||
auto skipped = skipTextSelection(selection);
|
auto skipped = skipTextSelection(selection);
|
||||||
auto mediaDisplayed = (_media && _media->isDisplayed());
|
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>()) {
|
if (auto entry = Get<HistoryMessageLogEntryOriginal>()) {
|
||||||
const auto originalSelection = mediaDisplayed
|
const auto originalSelection = mediaDisplayed
|
||||||
? _media->skipSelection(skipped)
|
? _media->skipSelection(skipped)
|
||||||
|
|
Loading…
Reference in New Issue