Compare commits

...

7 commits

Author SHA1 Message Date
Bruno Windels
2bba8f09ed WIP: styling for date separator 2022-06-25 12:52:44 +02:00
Bruno Windels
fa534b0ca9 add a binding here for the day we support edits 2022-06-25 12:52:26 +02:00
Bruno Windels
457c096c9b render date separator in base class for all tiles
apart from gap, which doesn't have date

since we add a container when the date separator needs to be shown, and
we don't want to rerender the whole tile, we always render timeline
tiles with a DIV rather than an LI (same for the UL).
2022-06-25 12:50:53 +02:00
Bruno Windels
a488ff143e also always update siblings when adding an entry,even when there is none
otherwise the first item in the list won't have a date separator
2022-06-25 12:48:14 +02:00
Bruno Windels
49752a0e7c implement logic when date separator should be shown 2022-06-25 12:47:57 +02:00
Bruno Windels
6697bb8ddf cleanup 2022-06-25 12:37:12 +02:00
Bruno Windels
e69a39d6a7 move date up to SimpleTile
we'll need it to see if we have a date separator
2022-06-25 12:36:39 +02:00
10 changed files with 127 additions and 49 deletions

View file

@ -130,15 +130,11 @@ export class TilesCollection extends BaseObservableList {
const newTile = this._createTile(entry);
if (newTile) {
if (prevTile) {
prevTile.updateNextSibling(newTile);
// this emits an update while the add hasn't been emitted yet
newTile.updatePreviousSibling(prevTile);
}
if (nextTile) {
newTile.updateNextSibling(nextTile);
nextTile.updatePreviousSibling(newTile);
}
prevTile?.updateNextSibling(newTile);
// this emits an update while the add hasn't been emitted yet
newTile.updatePreviousSibling(prevTile);
newTile.updateNextSibling(nextTile);
nextTile?.updatePreviousSibling(newTile);
this._tiles.splice(tileIdx, 0, newTile);
this.emitAdd(tileIdx, newTile);
// add event is emitted, now the tile

View file

@ -21,7 +21,6 @@ import {getIdentifierColorNumber, avatarInitials, getAvatarHttpUrl} from "../../
export class BaseMessageTile extends SimpleTile {
constructor(entry, options) {
super(entry, options);
this._date = this._entry.timestamp ? new Date(this._entry.timestamp) : null;
this._isContinuation = false;
this._reactions = null;
this._replyTile = null;
@ -78,10 +77,6 @@ export class BaseMessageTile extends SimpleTile {
return this.displayName;
}
get date() {
return this._date && this._date.toLocaleDateString({}, {month: "numeric", day: "numeric"});
}
get time() {
return this._date && this._date.toLocaleTimeString({}, {hour: "numeric", minute: "2-digit"});
}
@ -138,8 +133,7 @@ export class BaseMessageTile extends SimpleTile {
const action = this._replyTile?.updateEntry(replyEntry, param);
if (action?.shouldReplace || !this._replyTile) {
this.disposeTracked(this._replyTile);
const tileClassForEntry = this._options.tileClassForEntry;
const ReplyTile = tileClassForEntry(replyEntry);
const ReplyTile = this._options.tileClassForEntry(replyEntry);
if (ReplyTile) {
this._replyTile = new ReplyTile(replyEntry, this._options);
}

View file

@ -22,6 +22,8 @@ export class SimpleTile extends ViewModel {
constructor(entry, options) {
super(options);
this._entry = entry;
this._date = this._entry.timestamp ? new Date(this._entry.timestamp) : null;
this._hasDateSeparator = false;
this._emitUpdate = undefined;
}
// view model props for all subclasses
@ -38,13 +40,35 @@ export class SimpleTile extends ViewModel {
}
get hasDateSeparator() {
return false;
return this._hasDateSeparator;
}
_updateDateSeparator(prev) {
let hasDateSeparator;
if (prev instanceof SimpleTile) {
if (prev && prev._date) {
hasDateSeparator = prev._date.getFullYear() !== this._date.getFullYear() ||
prev._date.getMonth() !== this._date.getMonth() ||
prev._date.getDate() !== this._date.getDate();
} else {
hasDateSeparator = !!this._date;
}
} else {
hasDateSeparator = true;
}
const changed = hasDateSeparator !== this._hasDateSeparator;
this._hasDateSeparator = hasDateSeparator;
return changed;
}
get id() {
return this._entry.asEventKey();
}
get date() {
return this._date && this._date.toLocaleDateString({}, {weekday: 'long', year: 'numeric', month: 'long', day: 'numeric'});
}
get eventId() {
return this._entry.id;
}
@ -122,9 +146,12 @@ export class SimpleTile extends ViewModel {
tryIncludeEntry() {
return false;
}
// let item know it has a new sibling
updatePreviousSibling(/*prev*/) {
// let item know it has a new sibling
updatePreviousSibling(prev) {
if (this._updateDateSeparator(prev)) {
this.emitChange();
}
}
// let item know it has a new sibling

View file

@ -103,7 +103,7 @@ limitations under the License.
/* reset body margin */
margin: 0;
/* first try break-all, then break-word, which isn't supported everywhere */
word-break: break-all;
word-break: break-all;
word-break: break-word;
}
@ -422,3 +422,17 @@ only loads when the top comes into view*/
.GapView.isAtTop {
padding: 52px 20px 12px 20px;
}
.DateSeparator {
display: flex;
justify-content: center;
margin: 8px 0;
}
.DateSeparator > time {
background-color: var(--background-color-primary--darker-10);
border-radius: 16px;
padding: 4px 16px;
}

View file

@ -55,7 +55,7 @@ export class MessageComposer extends TemplateView {
className: "cancel",
onClick: () => this._clearReplyingTo()
}, "Close"),
t.view(new TileView(rvm, this._viewClassForTile, { interactive: false }, "div"))
t.view(new TileView(rvm, this._viewClassForTile, { interactive: false }))
]);
});
const input = t.div({className: "MessageComposer_input"}, [

View file

@ -192,6 +192,7 @@ class TilesListView extends ListView<SimpleTile, TileView> {
super({
list: tiles,
onItemClick: (tileView, evt) => tileView.onClick(evt),
tagName: "div"
}, tile => {
const TileView = viewClassForTile(tile);
return new TileView(tile, viewClassForTile);

View file

@ -14,18 +14,10 @@ See the License for the specific language governing permissions and
limitations under the License.
*/
import {TemplateView} from "../../../general/TemplateView";
import {BaseTileView} from "./BaseTileView";
export class AnnouncementView extends TemplateView {
// ignore other arguments
constructor(vm) {
super(vm);
export class AnnouncementView extends BaseTileView {
renderTile(t, vm) {
return t.div({className: "AnnouncementView"}, t.div(vm => vm.announcement));
}
render(t) {
return t.li({className: "AnnouncementView"}, t.div(vm => vm.announcement));
}
/* This is called by the parent ListView, which just has 1 listener for the whole list */
onClick() {}
}

View file

@ -18,17 +18,15 @@ limitations under the License.
import {renderStaticAvatar} from "../../../avatar";
import {tag} from "../../../general/html";
import {mountView} from "../../../general/utils";
import {TemplateView} from "../../../general/TemplateView";
import {BaseTileView} from "./BaseTileView";
import {Popup} from "../../../general/Popup.js";
import {Menu} from "../../../general/Menu.js";
import {ReactionsView} from "./ReactionsView.js";
export class BaseMessageView extends TemplateView {
constructor(value, viewClassForTile, renderFlags, tagName = "li") {
super(value);
export class BaseMessageView extends BaseTileView {
constructor(value, viewClassForTile, renderFlags) {
super(value, viewClassForTile);
this._menuPopup = null;
this._tagName = tagName;
this._viewClassForTile = viewClassForTile;
// TODO An enum could be nice to make code easier to read at call sites.
this._renderFlags = renderFlags;
}
@ -36,12 +34,12 @@ export class BaseMessageView extends TemplateView {
get _interactive() { return this._renderFlags?.interactive ?? true; }
get _isReplyPreview() { return this._renderFlags?.reply; }
render(t, vm) {
renderTile(t, vm) {
const children = [this.renderMessageBody(t, vm)];
if (this._interactive) {
children.push(t.button({className: "Timeline_messageOptions"}, "⋯"));
}
const li = t.el(this._tagName, {
const tile = t.div({
className: {
"Timeline_message": true,
own: vm.isOwn,
@ -59,13 +57,13 @@ export class BaseMessageView extends TemplateView {
// don't use `t` from within the side-effect callback
t.mapSideEffect(vm => vm.isContinuation, (isContinuation, wasContinuation) => {
if (isContinuation && wasContinuation === false) {
li.removeChild(li.querySelector(".Timeline_messageAvatar"));
li.removeChild(li.querySelector(".Timeline_messageSender"));
tile.removeChild(tile.querySelector(".Timeline_messageAvatar"));
tile.removeChild(tile.querySelector(".Timeline_messageSender"));
} else if (!isContinuation && !this._isReplyPreview) {
const avatar = tag.a({href: vm.memberPanelLink, className: "Timeline_messageAvatar"}, [renderStaticAvatar(vm, 30)]);
const sender = tag.div({className: `Timeline_messageSender usercolor${vm.avatarColorNumber}`}, vm.displayName);
li.insertBefore(avatar, li.firstChild);
li.insertBefore(sender, li.firstChild);
tile.insertBefore(avatar, tile.firstChild);
tile.insertBefore(sender, tile.firstChild);
}
});
// similarly, we could do this with a simple ifView,
@ -75,15 +73,15 @@ export class BaseMessageView extends TemplateView {
if (reactions && this._interactive && !reactionsView) {
reactionsView = new ReactionsView(reactions);
this.addSubView(reactionsView);
li.appendChild(mountView(reactionsView));
tile.appendChild(mountView(reactionsView));
} else if (!reactions && reactionsView) {
li.removeChild(reactionsView.root());
tile.removeChild(reactionsView.root());
reactionsView.unmount();
this.removeSubView(reactionsView);
reactionsView = null;
}
});
return li;
return tile;
}
/* This is called by the parent ListView, which just has 1 listener for the whole list */

View file

@ -0,0 +1,56 @@
/*
Copyright 2020 Bruno Windels <bruno@windels.cloud>
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
import {TemplateView} from "../../../general/TemplateView";
export class BaseTileView extends TemplateView {
// ignore other arguments
constructor(vm, viewClassForTile) {
super(vm);
this._viewClassForTile = viewClassForTile;
this._root = undefined;
}
root() {
return this._root;
}
render(t, vm) {
const tile = this.renderTile(t, vm);
const swapRoot = newRoot => {
this._root?.replaceWith(newRoot);
this._root = newRoot;
}
t.mapSideEffect(vm => vm.hasDateSeparator, hasDateSeparator => {
if (hasDateSeparator) {
const container = t.div([this._renderDateSeparator(t, vm)]);
swapRoot(container);
container.appendChild(tile);
} else {
swapRoot(tile);
}
});
return this._root;
}
_renderDateSeparator(t, vm) {
// if this needs any bindings, we need to use a subview
return t.div({className: "DateSeparator"}, t.time(vm.date));
}
/* This is called by the parent ListView, which just has 1 listener for the whole list */
onClick() {}
}

View file

@ -20,7 +20,7 @@ import {ReplyPreviewError, ReplyPreviewView} from "./ReplyPreviewView.js";
export class TextMessageView extends BaseMessageView {
renderMessageBody(t, vm) {
const time = t.time({className: {hidden: !vm.date}}, vm.date + " " + vm.time);
const time = t.time({className: {hidden: vm => !vm.time}}, vm => vm.time);
const container = t.div({
className: {
"Timeline_messageBody": true,