mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-19 22:51:41 +08:00
256 lines
7.8 KiB
Markdown
256 lines
7.8 KiB
Markdown
|
[#]: collector: (lujun9972)
|
|||
|
[#]: translator: ( )
|
|||
|
[#]: reviewer: ( )
|
|||
|
[#]: publisher: ( )
|
|||
|
[#]: url: ( )
|
|||
|
[#]: subject: (Building a Messenger App: Home Page)
|
|||
|
[#]: via: (https://nicolasparada.netlify.com/posts/go-messenger-home-page/)
|
|||
|
[#]: author: (Nicolás Parada https://nicolasparada.netlify.com/)
|
|||
|
|
|||
|
Building a Messenger App: Home Page
|
|||
|
======
|
|||
|
|
|||
|
This post is the 8th on a series:
|
|||
|
|
|||
|
* [Part 1: Schema][1]
|
|||
|
* [Part 2: OAuth][2]
|
|||
|
* [Part 3: Conversations][3]
|
|||
|
* [Part 4: Messages][4]
|
|||
|
* [Part 5: Realtime Messages][5]
|
|||
|
* [Part 6: Development Login][6]
|
|||
|
* [Part 7: Access Page][7]
|
|||
|
|
|||
|
|
|||
|
|
|||
|
Continuing the frontend, let’s finish the home page in this post. We’ll add a form to start conversations and a list with the latest ones.
|
|||
|
|
|||
|
### Conversation Form
|
|||
|
|
|||
|
![conversation form screenshot][8]
|
|||
|
|
|||
|
In the `static/pages/home-page.js` file add some markup in the HTML view.
|
|||
|
|
|||
|
```
|
|||
|
<form id="conversation-form">
|
|||
|
<input type="search" placeholder="Start conversation with..." required>
|
|||
|
</form>
|
|||
|
```
|
|||
|
|
|||
|
Add that form just below the section in which we displayed the auth user and logout button.
|
|||
|
|
|||
|
```
|
|||
|
page.getElementById('conversation-form').onsubmit = onConversationSubmit
|
|||
|
```
|
|||
|
|
|||
|
Now we can listen to the “submit” event to create the conversation.
|
|||
|
|
|||
|
```
|
|||
|
import http from '../http.js'
|
|||
|
import { navigate } from '../router.js'
|
|||
|
|
|||
|
async function onConversationSubmit(ev) {
|
|||
|
ev.preventDefault()
|
|||
|
|
|||
|
const form = ev.currentTarget
|
|||
|
const input = form.querySelector('input')
|
|||
|
|
|||
|
input.disabled = true
|
|||
|
|
|||
|
try {
|
|||
|
const conversation = await createConversation(input.value)
|
|||
|
input.value = ''
|
|||
|
navigate('/conversations/' + conversation.id)
|
|||
|
} catch (err) {
|
|||
|
if (err.statusCode === 422) {
|
|||
|
input.setCustomValidity(err.body.errors.username)
|
|||
|
} else {
|
|||
|
alert(err.message)
|
|||
|
}
|
|||
|
setTimeout(() => {
|
|||
|
input.focus()
|
|||
|
}, 0)
|
|||
|
} finally {
|
|||
|
input.disabled = false
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
function createConversation(username) {
|
|||
|
return http.post('/api/conversations', { username })
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
On submit we do a POST request to `/api/conversations` with the username and redirect to the conversation page (for the next post).
|
|||
|
|
|||
|
### Conversation List
|
|||
|
|
|||
|
![conversation list screenshot][9]
|
|||
|
|
|||
|
In the same file, we are going to make the `homePage()` function async to load the conversations first.
|
|||
|
|
|||
|
```
|
|||
|
export default async function homePage() {
|
|||
|
const conversations = await getConversations().catch(err => {
|
|||
|
console.error(err)
|
|||
|
return []
|
|||
|
})
|
|||
|
/*...*/
|
|||
|
}
|
|||
|
|
|||
|
function getConversations() {
|
|||
|
return http.get('/api/conversations')
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
Then, add a list in the markup to render conversations there.
|
|||
|
|
|||
|
```
|
|||
|
<ol id="conversations"></ol>
|
|||
|
```
|
|||
|
|
|||
|
Add it just below the current markup.
|
|||
|
|
|||
|
```
|
|||
|
const conversationsOList = page.getElementById('conversations')
|
|||
|
for (const conversation of conversations) {
|
|||
|
conversationsOList.appendChild(renderConversation(conversation))
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
So we can append each conversation to the list.
|
|||
|
|
|||
|
```
|
|||
|
import { avatar, escapeHTML } from '../shared.js'
|
|||
|
|
|||
|
function renderConversation(conversation) {
|
|||
|
const messageContent = escapeHTML(conversation.lastMessage.content)
|
|||
|
const messageDate = new Date(conversation.lastMessage.createdAt).toLocaleString()
|
|||
|
|
|||
|
const li = document.createElement('li')
|
|||
|
li.dataset['id'] = conversation.id
|
|||
|
if (conversation.hasUnreadMessages) {
|
|||
|
li.classList.add('has-unread-messages')
|
|||
|
}
|
|||
|
li.innerHTML = `
|
|||
|
<a href="/conversations/${conversation.id}">
|
|||
|
<div>
|
|||
|
${avatar(conversation.otherParticipant)}
|
|||
|
<span>${conversation.otherParticipant.username}</span>
|
|||
|
</div>
|
|||
|
<div>
|
|||
|
<p>${messageContent}</p>
|
|||
|
<time>${messageDate}</time>
|
|||
|
</div>
|
|||
|
</a>
|
|||
|
`
|
|||
|
return li
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
Each conversation item contains a link to the conversation page and displays the other participant info and a preview of the last message. Also, you can use `.hasUnreadMessages` to add a class to the item and do some styling with CSS. Maybe a bolder font or accent the color.
|
|||
|
|
|||
|
Note that we’re escaping the message content. That function comes from `static/shared.js`:
|
|||
|
|
|||
|
```
|
|||
|
export function escapeHTML(str) {
|
|||
|
return str
|
|||
|
.replace(/&/g, '&')
|
|||
|
.replace(/</g, '<')
|
|||
|
.replace(/>/g, '>')
|
|||
|
.replace(/"/g, '"')
|
|||
|
.replace(/'/g, ''')
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
That prevents displaying as HTML the message the user wrote. If the user happens to write something like:
|
|||
|
|
|||
|
```
|
|||
|
<script>alert('lololo')</script>
|
|||
|
```
|
|||
|
|
|||
|
It would be very annoying because that script will be executed 😅
|
|||
|
So yeah, always remember to escape content from untrusted sources.
|
|||
|
|
|||
|
### Messages Subscription
|
|||
|
|
|||
|
Last but not least, I want to subscribe to the message stream here.
|
|||
|
|
|||
|
```
|
|||
|
const unsubscribe = subscribeToMessages(onMessageArrive)
|
|||
|
page.addEventListener('disconnect', unsubscribe)
|
|||
|
```
|
|||
|
|
|||
|
Add that line in the `homePage()` function.
|
|||
|
|
|||
|
```
|
|||
|
function subscribeToMessages(cb) {
|
|||
|
return http.subscribe('/api/messages', cb)
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
The `subscribe()` function returns a function that once called it closes the underlying connection. That’s why I passed it to the “disconnect” event; so when the user leaves the page, the event stream will be closed.
|
|||
|
|
|||
|
```
|
|||
|
async function onMessageArrive(message) {
|
|||
|
const conversationLI = document.querySelector(`li[data-id="${message.conversationID}"]`)
|
|||
|
if (conversationLI !== null) {
|
|||
|
conversationLI.classList.add('has-unread-messages')
|
|||
|
conversationLI.querySelector('a > div > p').textContent = message.content
|
|||
|
conversationLI.querySelector('a > div > time').textContent = new Date(message.createdAt).toLocaleString()
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
let conversation
|
|||
|
try {
|
|||
|
conversation = await getConversation(message.conversationID)
|
|||
|
conversation.lastMessage = message
|
|||
|
} catch (err) {
|
|||
|
console.error(err)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
const conversationsOList = document.getElementById('conversations')
|
|||
|
if (conversationsOList === null) {
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
conversationsOList.insertAdjacentElement('afterbegin', renderConversation(conversation))
|
|||
|
}
|
|||
|
|
|||
|
function getConversation(id) {
|
|||
|
return http.get('/api/conversations/' + id)
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
Every time a new message arrives, we go and query for the conversation item in the DOM. If found, we add the `has-unread-messages` class to the item, and update the view. If not found, it means the message is from a new conversation created just now. We go and do a GET request to `/api/conversations/{conversationID}` to get the conversation in which the message was created and prepend it to the conversation list.
|
|||
|
|
|||
|
* * *
|
|||
|
|
|||
|
That covers the home page 😊
|
|||
|
On the next post we’ll code the conversation page.
|
|||
|
|
|||
|
[Souce Code][10]
|
|||
|
|
|||
|
--------------------------------------------------------------------------------
|
|||
|
|
|||
|
via: https://nicolasparada.netlify.com/posts/go-messenger-home-page/
|
|||
|
|
|||
|
作者:[Nicolás Parada][a]
|
|||
|
选题:[lujun9972][b]
|
|||
|
译者:[译者ID](https://github.com/译者ID)
|
|||
|
校对:[校对者ID](https://github.com/校对者ID)
|
|||
|
|
|||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
|||
|
|
|||
|
[a]: https://nicolasparada.netlify.com/
|
|||
|
[b]: https://github.com/lujun9972
|
|||
|
[1]: https://nicolasparada.netlify.com/posts/go-messenger-schema/
|
|||
|
[2]: https://nicolasparada.netlify.com/posts/go-messenger-oauth/
|
|||
|
[3]: https://nicolasparada.netlify.com/posts/go-messenger-conversations/
|
|||
|
[4]: https://nicolasparada.netlify.com/posts/go-messenger-messages/
|
|||
|
[5]: https://nicolasparada.netlify.com/posts/go-messenger-realtime-messages/
|
|||
|
[6]: https://nicolasparada.netlify.com/posts/go-messenger-dev-login/
|
|||
|
[7]: https://nicolasparada.netlify.com/posts/go-messenger-access-page/
|
|||
|
[8]: https://nicolasparada.netlify.com/img/go-messenger-home-page/conversation-form.png
|
|||
|
[9]: https://nicolasparada.netlify.com/img/go-messenger-home-page/conversation-list.png
|
|||
|
[10]: https://github.com/nicolasparada/go-messenger-demo
|