mirror of
				https://github.com/vector-im/element-web.git
				synced 2025-10-24 22:01:46 +02:00 
			
		
		
		
	Improve reply rendering
Signed-off-by: Tulir Asokan <tulir@maunium.net>
This commit is contained in:
		
							parent
							
								
									385e83fdbc
								
							
						
					
					
						commit
						d282675bc6
					
				| @ -153,6 +153,7 @@ | |||||||
| @import "./views/rooms/_PinnedEventsPanel.scss"; | @import "./views/rooms/_PinnedEventsPanel.scss"; | ||||||
| @import "./views/rooms/_PresenceLabel.scss"; | @import "./views/rooms/_PresenceLabel.scss"; | ||||||
| @import "./views/rooms/_ReplyPreview.scss"; | @import "./views/rooms/_ReplyPreview.scss"; | ||||||
|  | @import "./views/rooms/_ReplyTile.scss"; | ||||||
| @import "./views/rooms/_RoomBreadcrumbs.scss"; | @import "./views/rooms/_RoomBreadcrumbs.scss"; | ||||||
| @import "./views/rooms/_RoomDropTarget.scss"; | @import "./views/rooms/_RoomDropTarget.scss"; | ||||||
| @import "./views/rooms/_RoomHeader.scss"; | @import "./views/rooms/_RoomHeader.scss"; | ||||||
|  | |||||||
| @ -18,20 +18,13 @@ limitations under the License. | |||||||
|     margin-top: 0; |     margin-top: 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mx_ReplyThread .mx_DateSeparator { |  | ||||||
|     font-size: 1em !important; |  | ||||||
|     margin-top: 0; |  | ||||||
|     margin-bottom: 0; |  | ||||||
|     padding-bottom: 1px; |  | ||||||
|     bottom: -5px; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .mx_ReplyThread_show { | .mx_ReplyThread_show { | ||||||
|     cursor: pointer; |     cursor: pointer; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| blockquote.mx_ReplyThread { | blockquote.mx_ReplyThread { | ||||||
|     margin-left: 0; |     margin-left: 0; | ||||||
|  |     margin-bottom: 8px; | ||||||
|     padding-left: 10px; |     padding-left: 10px; | ||||||
|     border-left: 4px solid $blockquote-bar-color; |     border-left: 4px solid $button-bg-color; | ||||||
| } | } | ||||||
|  | |||||||
| @ -32,12 +32,16 @@ limitations under the License. | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mx_ReplyPreview_header { | .mx_ReplyPreview_header { | ||||||
|     margin: 12px; |     margin: 8px; | ||||||
|     color: $primary-fg-color; |     color: $primary-fg-color; | ||||||
|     font-weight: 400; |     font-weight: 400; | ||||||
|     opacity: 0.4; |     opacity: 0.4; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .mx_ReplyPreview_tile { | ||||||
|  |     margin: 0 8px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .mx_ReplyPreview_title { | .mx_ReplyPreview_title { | ||||||
|     float: left; |     float: left; | ||||||
| } | } | ||||||
| @ -45,6 +49,7 @@ limitations under the License. | |||||||
| .mx_ReplyPreview_cancel { | .mx_ReplyPreview_cancel { | ||||||
|     float: right; |     float: right; | ||||||
|     cursor: pointer; |     cursor: pointer; | ||||||
|  |     display: flex; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mx_ReplyPreview_clear { | .mx_ReplyPreview_clear { | ||||||
|  | |||||||
							
								
								
									
										96
									
								
								res/css/views/rooms/_ReplyTile.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								res/css/views/rooms/_ReplyTile.scss
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,96 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2019 Tulir Asokan <tulir@maunium.net> | ||||||
|  | 
 | ||||||
|  | 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. | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile { | ||||||
|  |     max-width: 100%; | ||||||
|  |     clear: both; | ||||||
|  |     padding-top: 2px; | ||||||
|  |     padding-bottom: 2px; | ||||||
|  |     font-size: 14px; | ||||||
|  |     position: relative; | ||||||
|  |     line-height: 16px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile > a { | ||||||
|  |     display: block; | ||||||
|  |     text-decoration: none; | ||||||
|  |     color: $primary-fg-color; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // We do reply size limiting with CSS to avoid duplicating the TextualBody component. | ||||||
|  | .mx_ReplyTile .mx_EventTile_content { | ||||||
|  |     $reply-lines: 2; | ||||||
|  |     $line-height: 22px; | ||||||
|  | 
 | ||||||
|  |     text-overflow: ellipsis; | ||||||
|  |     display: -webkit-box; | ||||||
|  |     -webkit-box-orient: vertical; | ||||||
|  |     -webkit-line-clamp: $reply-lines; | ||||||
|  |     line-height: $line-height; | ||||||
|  | 
 | ||||||
|  |     .mx_EventTile_body.mx_EventTile_bigEmoji { | ||||||
|  |         line-height: $line-height !important; | ||||||
|  |         // Override the big emoji override | ||||||
|  |         font-size: 14px !important; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile.mx_ReplyTile_info { | ||||||
|  |     padding-top: 0px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile .mx_SenderProfile { | ||||||
|  |     color: $primary-fg-color; | ||||||
|  |     font-size: 14px; | ||||||
|  |     display: inline-block; /* anti-zalgo, with overflow hidden */ | ||||||
|  |     overflow: hidden; | ||||||
|  |     cursor: pointer; | ||||||
|  |     padding-left: 0px; /* left gutter */ | ||||||
|  |     padding-bottom: 0px; | ||||||
|  |     padding-top: 0px; | ||||||
|  |     margin: 0px; | ||||||
|  |     line-height: 17px; | ||||||
|  |     /* the next three lines, along with overflow hidden, truncate long display names */ | ||||||
|  |     white-space: nowrap; | ||||||
|  |     text-overflow: ellipsis; | ||||||
|  |     max-width: calc(100% - 65px); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile_redacted .mx_UnknownBody { | ||||||
|  |     --lozenge-color: $event-redacted-fg-color; | ||||||
|  |     --lozenge-border-color: $event-redacted-border-color; | ||||||
|  |     display: block; | ||||||
|  |     height: 22px; | ||||||
|  |     width: 250px; | ||||||
|  |     border-radius: 11px; | ||||||
|  |     background: | ||||||
|  |         repeating-linear-gradient( | ||||||
|  |             -45deg, | ||||||
|  |             var(--lozenge-color), | ||||||
|  |             var(--lozenge-color) 3px, | ||||||
|  |             transparent 3px, | ||||||
|  |             transparent 6px | ||||||
|  |         ); | ||||||
|  |     box-shadow: 0px 0px 3px var(--lozenge-border-color) inset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile_sending.mx_ReplyTile_redacted .mx_UnknownBody { | ||||||
|  |     opacity: 0.4; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mx_ReplyTile_contextual { | ||||||
|  |     opacity: 0.4; | ||||||
|  | } | ||||||
| @ -304,20 +304,11 @@ export default class ReplyThread extends React.Component { | |||||||
|             header = <Spinner w={16} h={16} />; |             header = <Spinner w={16} h={16} />; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         const EventTile = sdk.getComponent('views.rooms.EventTile'); |         const ReplyTile = sdk.getComponent('views.rooms.ReplyTile'); | ||||||
|         const DateSeparator = sdk.getComponent('messages.DateSeparator'); |  | ||||||
|         const evTiles = this.state.events.map((ev) => { |         const evTiles = this.state.events.map((ev) => { | ||||||
|             let dateSep = null; |  | ||||||
| 
 |  | ||||||
|             if (wantsDateSeparator(this.props.parentEv.getDate(), ev.getDate())) { |  | ||||||
|                 dateSep = <a href={this.props.url}><DateSeparator ts={ev.getTs()} /></a>; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return <blockquote className="mx_ReplyThread" key={ev.getId()}> |             return <blockquote className="mx_ReplyThread" key={ev.getId()}> | ||||||
|                 { dateSep } |                 <ReplyTile | ||||||
|                 <EventTile |  | ||||||
|                     mxEvent={ev} |                     mxEvent={ev} | ||||||
|                     tileShape="reply" |  | ||||||
|                     onHeightChanged={this.props.onHeightChanged} |                     onHeightChanged={this.props.onHeightChanged} | ||||||
|                     permalinkCreator={this.props.permalinkCreator} |                     permalinkCreator={this.props.permalinkCreator} | ||||||
|                     isRedacted={ev.isRedacted()} |                     isRedacted={ev.isRedacted()} | ||||||
| @ -325,7 +316,7 @@ export default class ReplyThread extends React.Component { | |||||||
|             </blockquote>; |             </blockquote>; | ||||||
|         }); |         }); | ||||||
| 
 | 
 | ||||||
|         return <div> |         return <div className="mx_ReplyThread_wrapper"> | ||||||
|             <div>{ header }</div> |             <div>{ header }</div> | ||||||
|             <div>{ evTiles }</div> |             <div>{ evTiles }</div> | ||||||
|         </div>; |         </div>; | ||||||
|  | |||||||
							
								
								
									
										33
									
								
								src/components/views/messages/MImageReplyBody.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								src/components/views/messages/MImageReplyBody.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2019 Tulir Asokan <tulir@maunium.net> | ||||||
|  | 
 | ||||||
|  | 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 React from 'react'; | ||||||
|  | import MImageBody from './MImageBody'; | ||||||
|  | 
 | ||||||
|  | export default class MImageReplyBody extends MImageBody { | ||||||
|  |     onClick(ev) { | ||||||
|  |         ev.preventDefault(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     wrapImage(contentUrl, children) { | ||||||
|  |         return children; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Don't show "Download this_file.png ..."
 | ||||||
|  |     getFileBody() { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -43,6 +43,9 @@ module.exports = createReactClass({ | |||||||
| 
 | 
 | ||||||
|         /* the maximum image height to use, if the event is an image */ |         /* the maximum image height to use, if the event is an image */ | ||||||
|         maxImageHeight: PropTypes.number, |         maxImageHeight: PropTypes.number, | ||||||
|  | 
 | ||||||
|  |         overrideBodyTypes: PropTypes.object, | ||||||
|  |         overrideEventTypes: PropTypes.object, | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     getEventTileOps: function() { |     getEventTileOps: function() { | ||||||
| @ -60,9 +63,11 @@ module.exports = createReactClass({ | |||||||
|             'm.file': sdk.getComponent('messages.MFileBody'), |             'm.file': sdk.getComponent('messages.MFileBody'), | ||||||
|             'm.audio': sdk.getComponent('messages.MAudioBody'), |             'm.audio': sdk.getComponent('messages.MAudioBody'), | ||||||
|             'm.video': sdk.getComponent('messages.MVideoBody'), |             'm.video': sdk.getComponent('messages.MVideoBody'), | ||||||
|  |             ...(this.props.overrideBodyTypes || {}), | ||||||
|         }; |         }; | ||||||
|         const evTypes = { |         const evTypes = { | ||||||
|             'm.sticker': sdk.getComponent('messages.MStickerBody'), |             'm.sticker': sdk.getComponent('messages.MStickerBody'), | ||||||
|  |             ...(this.props.overrideEventTypes || {}), | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         const content = this.props.mxEvent.getContent(); |         const content = this.props.mxEvent.getContent(); | ||||||
| @ -81,7 +86,7 @@ module.exports = createReactClass({ | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return <BodyType |         return BodyType ? <BodyType | ||||||
|             ref="body" mxEvent={this.props.mxEvent} |             ref="body" mxEvent={this.props.mxEvent} | ||||||
|             highlights={this.props.highlights} |             highlights={this.props.highlights} | ||||||
|             highlightLink={this.props.highlightLink} |             highlightLink={this.props.highlightLink} | ||||||
| @ -90,6 +95,6 @@ module.exports = createReactClass({ | |||||||
|             maxImageHeight={this.props.maxImageHeight} |             maxImageHeight={this.props.maxImageHeight} | ||||||
|             replacingEventId={this.props.replacingEventId} |             replacingEventId={this.props.replacingEventId} | ||||||
|             editState={this.props.editState} |             editState={this.props.editState} | ||||||
|             onHeightChanged={this.props.onHeightChanged} />; |             onHeightChanged={this.props.onHeightChanged} /> : null; | ||||||
|     }, |     }, | ||||||
| }); | }); | ||||||
|  | |||||||
| @ -68,7 +68,7 @@ export default class ReplyPreview extends React.Component { | |||||||
|     render() { |     render() { | ||||||
|         if (!this.state.event) return null; |         if (!this.state.event) return null; | ||||||
| 
 | 
 | ||||||
|         const EventTile = sdk.getComponent('rooms.EventTile'); |         const ReplyTile = sdk.getComponent('rooms.ReplyTile'); | ||||||
| 
 | 
 | ||||||
|         return <div className="mx_ReplyPreview"> |         return <div className="mx_ReplyPreview"> | ||||||
|             <div className="mx_ReplyPreview_section"> |             <div className="mx_ReplyPreview_section"> | ||||||
| @ -80,11 +80,11 @@ export default class ReplyPreview extends React.Component { | |||||||
|                          onClick={cancelQuoting} /> |                          onClick={cancelQuoting} /> | ||||||
|                 </div> |                 </div> | ||||||
|                 <div className="mx_ReplyPreview_clear" /> |                 <div className="mx_ReplyPreview_clear" /> | ||||||
|                 <EventTile last={true} |                 <div className="mx_ReplyPreview_tile"> | ||||||
|                            tileShape="reply_preview" |                     <ReplyTile isRedacted={this.state.event.isRedacted()} | ||||||
|                            mxEvent={this.state.event} |                                mxEvent={this.state.event} | ||||||
|                            permalinkCreator={this.props.permalinkCreator} |                                permalinkCreator={this.props.permalinkCreator} /> | ||||||
|                            isTwelveHour={SettingsStore.getValue("showTwelveHourTimestamps")} /> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|         </div>; |         </div>; | ||||||
|     } |     } | ||||||
|  | |||||||
							
								
								
									
										234
									
								
								src/components/views/rooms/ReplyTile.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										234
									
								
								src/components/views/rooms/ReplyTile.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,234 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2019 Tulir Asokan <tulir@maunium.net> | ||||||
|  | 
 | ||||||
|  | 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 React from 'react'; | ||||||
|  | import PropTypes from 'prop-types'; | ||||||
|  | const classNames = require("classnames"); | ||||||
|  | import { _t, _td } from '../../../languageHandler'; | ||||||
|  | 
 | ||||||
|  | const sdk = require('../../../index'); | ||||||
|  | 
 | ||||||
|  | import dis from '../../../dispatcher'; | ||||||
|  | import SettingsStore from "../../../settings/SettingsStore"; | ||||||
|  | import {MatrixClient} from 'matrix-js-sdk'; | ||||||
|  | 
 | ||||||
|  | const ObjectUtils = require('../../../ObjectUtils'); | ||||||
|  | 
 | ||||||
|  | const eventTileTypes = { | ||||||
|  |     'm.room.message': 'messages.MessageEvent', | ||||||
|  |     'm.sticker': 'messages.MessageEvent', | ||||||
|  |     'm.call.invite': 'messages.TextualEvent', | ||||||
|  |     'm.call.answer': 'messages.TextualEvent', | ||||||
|  |     'm.call.hangup': 'messages.TextualEvent', | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const stateEventTileTypes = { | ||||||
|  |     'm.room.aliases': 'messages.TextualEvent', | ||||||
|  |     // 'm.room.aliases': 'messages.RoomAliasesEvent', // too complex
 | ||||||
|  |     'm.room.canonical_alias': 'messages.TextualEvent', | ||||||
|  |     'm.room.create': 'messages.RoomCreate', | ||||||
|  |     'm.room.member': 'messages.TextualEvent', | ||||||
|  |     'm.room.name': 'messages.TextualEvent', | ||||||
|  |     'm.room.avatar': 'messages.RoomAvatarEvent', | ||||||
|  |     'm.room.third_party_invite': 'messages.TextualEvent', | ||||||
|  |     'm.room.history_visibility': 'messages.TextualEvent', | ||||||
|  |     'm.room.encryption': 'messages.TextualEvent', | ||||||
|  |     'm.room.topic': 'messages.TextualEvent', | ||||||
|  |     'm.room.power_levels': 'messages.TextualEvent', | ||||||
|  |     'm.room.pinned_events': 'messages.TextualEvent', | ||||||
|  |     'm.room.server_acl': 'messages.TextualEvent', | ||||||
|  |     'im.vector.modular.widgets': 'messages.TextualEvent', | ||||||
|  |     'm.room.tombstone': 'messages.TextualEvent', | ||||||
|  |     'm.room.join_rules': 'messages.TextualEvent', | ||||||
|  |     'm.room.guest_access': 'messages.TextualEvent', | ||||||
|  |     'm.room.related_groups': 'messages.TextualEvent', | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | function getHandlerTile(ev) { | ||||||
|  |     const type = ev.getType(); | ||||||
|  |     return ev.isState() ? stateEventTileTypes[type] : eventTileTypes[type]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | class ReplyTile extends React.Component { | ||||||
|  |     static contextTypes = { | ||||||
|  |         matrixClient: PropTypes.instanceOf(MatrixClient).isRequired, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static propTypes = { | ||||||
|  |         mxEvent: PropTypes.object.isRequired, | ||||||
|  |         isRedacted: PropTypes.bool, | ||||||
|  |         permalinkCreator: PropTypes.object, | ||||||
|  |         onHeightChanged: PropTypes.func, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     static defaultProps = { | ||||||
|  |         onHeightChanged: function() {}, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     constructor(props, context) { | ||||||
|  |         super(props, context); | ||||||
|  |         this.state = {}; | ||||||
|  |         this.onClick = this.onClick.bind(this); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     componentDidMount() { | ||||||
|  |         this.props.mxEvent.on("Event.decrypted", this._onDecrypted); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     shouldComponentUpdate(nextProps, nextState) { | ||||||
|  |         if (!ObjectUtils.shallowEqual(this.state, nextState)) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return !this._propsEqual(this.props, nextProps); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     componentWillUnmount() { | ||||||
|  |         const client = this.context.matrixClient; | ||||||
|  |         this.props.mxEvent.removeListener("Event.decrypted", this._onDecrypted); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     _onDecrypted() { | ||||||
|  |         this.forceUpdate(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     _propsEqual(objA, objB) { | ||||||
|  |         const keysA = Object.keys(objA); | ||||||
|  |         const keysB = Object.keys(objB); | ||||||
|  | 
 | ||||||
|  |         if (keysA.length !== keysB.length) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         for (let i = 0; i < keysA.length; i++) { | ||||||
|  |             const key = keysA[i]; | ||||||
|  | 
 | ||||||
|  |             if (!objB.hasOwnProperty(key)) { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (objA[key] !== objB[key]) { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     onClick(e) { | ||||||
|  |         // This allows the permalink to be opened in a new tab/window or copied as
 | ||||||
|  |         // matrix.to, but also for it to enable routing within Riot when clicked.
 | ||||||
|  |         e.preventDefault(); | ||||||
|  |         dis.dispatch({ | ||||||
|  |             action: 'view_room', | ||||||
|  |             event_id: this.props.mxEvent.getId(), | ||||||
|  |             highlighted: true, | ||||||
|  |             room_id: this.props.mxEvent.getRoomId(), | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     render() { | ||||||
|  |         const SenderProfile = sdk.getComponent('messages.SenderProfile'); | ||||||
|  | 
 | ||||||
|  |         const content = this.props.mxEvent.getContent(); | ||||||
|  |         const msgtype = content.msgtype; | ||||||
|  |         const eventType = this.props.mxEvent.getType(); | ||||||
|  | 
 | ||||||
|  |         // Info messages are basically information about commands processed on a room
 | ||||||
|  |         let isInfoMessage = ( | ||||||
|  |             eventType !== 'm.room.message' && eventType !== 'm.sticker' && eventType !== 'm.room.create' | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         let tileHandler = getHandlerTile(this.props.mxEvent); | ||||||
|  |         // If we're showing hidden events in the timeline, we should use the
 | ||||||
|  |         // source tile when there's no regular tile for an event and also for
 | ||||||
|  |         // replace relations (which otherwise would display as a confusing
 | ||||||
|  |         // duplicate of the thing they are replacing).
 | ||||||
|  |         const useSource = !tileHandler || this.props.mxEvent.isRelation("m.replace"); | ||||||
|  |         if (useSource && SettingsStore.getValue("showHiddenEventsInTimeline")) { | ||||||
|  |             tileHandler = "messages.ViewSourceEvent"; | ||||||
|  |             // Reuse info message avatar and sender profile styling
 | ||||||
|  |             isInfoMessage = true; | ||||||
|  |         } | ||||||
|  |         // This shouldn't happen: the caller should check we support this type
 | ||||||
|  |         // before trying to instantiate us
 | ||||||
|  |         if (!tileHandler) { | ||||||
|  |             const {mxEvent} = this.props; | ||||||
|  |             console.warn(`Event type not supported: type:${mxEvent.getType()} isState:${mxEvent.isState()}`); | ||||||
|  |             return <div className="mx_ReplyTile mx_ReplyTile_info mx_MNoticeBody"> | ||||||
|  |                 { _t('This event could not be displayed') } | ||||||
|  |             </div>; | ||||||
|  |         } | ||||||
|  |         const EventTileType = sdk.getComponent(tileHandler); | ||||||
|  | 
 | ||||||
|  |         const classes = classNames({ | ||||||
|  |             mx_ReplyTile: true, | ||||||
|  |             mx_ReplyTile_info: isInfoMessage, | ||||||
|  |             mx_ReplyTile_redacted: this.props.isRedacted, | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         let permalink = "#"; | ||||||
|  |         if (this.props.permalinkCreator) { | ||||||
|  |             permalink = this.props.permalinkCreator.forEvent(this.props.mxEvent.getId()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let sender; | ||||||
|  |         let needsSenderProfile = tileHandler !== 'messages.RoomCreate' && !isInfoMessage; | ||||||
|  | 
 | ||||||
|  |         if (needsSenderProfile) { | ||||||
|  |             let text = null; | ||||||
|  |             if (msgtype === 'm.image') text = _td('%(senderName)s sent an image'); | ||||||
|  |             else if (msgtype === 'm.video') text = _td('%(senderName)s sent a video'); | ||||||
|  |             else if (msgtype === 'm.file') text = _td('%(senderName)s uploaded a file'); | ||||||
|  |             sender = <SenderProfile onClick={this.onSenderProfileClick} | ||||||
|  |                 mxEvent={this.props.mxEvent} | ||||||
|  |                 enableFlair={false} | ||||||
|  |                 text={text} />; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const MImageReplyBody = sdk.getComponent('messages.MImageReplyBody'); | ||||||
|  |         const TextualBody = sdk.getComponent('messages.TextualBody'); | ||||||
|  |         const msgtypeOverrides = { | ||||||
|  |             "m.image": MImageReplyBody, | ||||||
|  |             // We don't want a download link for files, just the file name is enough.
 | ||||||
|  |             "m.file": TextualBody, | ||||||
|  |             "m.sticker": TextualBody, | ||||||
|  |             "m.audio": TextualBody, | ||||||
|  |             "m.video": TextualBody, | ||||||
|  |         }; | ||||||
|  |         const evOverrides = { | ||||||
|  |             "m.sticker": TextualBody, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         return ( | ||||||
|  |             <div className={classes}> | ||||||
|  |                 <a href={permalink} onClick={this.onClick}> | ||||||
|  |                     { sender } | ||||||
|  |                     <EventTileType ref="tile" | ||||||
|  |                         mxEvent={this.props.mxEvent} | ||||||
|  |                         highlights={this.props.highlights} | ||||||
|  |                         highlightLink={this.props.highlightLink} | ||||||
|  |                         onHeightChanged={this.props.onHeightChanged} | ||||||
|  |                         showUrlPreview={false} | ||||||
|  |                         overrideBodyTypes={msgtypeOverrides} | ||||||
|  |                         overrideEventTypes={evOverrides} | ||||||
|  |                         maxImageHeight={96}/> | ||||||
|  |                 </a> | ||||||
|  |             </div> | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = ReplyTile; | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user