<ul class="chat-thread">
<li>Are we meeting today?</li>
<li>yes, what time suits you?</li>
<li>I was thinking after lunch, I have a meeting in the morning</li>
</ul>
<form class="chat-window">
<input class="chat-window-message" name="chat-window-message" type="text" autocomplete="off" autofocus />
</form>
// Imports
// --------------------------------------
@import 'https://fonts.googleapis.com/css?family=Noto+Sans';
// Variables
// --------------------------------------
$scrollbar-width: 10px;
$chat-thread-bgd-color: rgba(25,147,147,0.2);
$chat-thread-msg-arrow-size: 15px;
$chat-thread-avatar-size: 50px;
$chat-thread-offset: #{$chat-thread-avatar-size + 30px};
body {
padding: 0;
margin: 0;
background: linear-gradient(-45deg, #183850 0, #183850 25%, #192C46 50%, #22254C 75%, #22254C 100%);
background: linear-gradient(-45deg, #183850 0, #183850 25%, #192C46 50%, #22254C 75%, #22254C 100%);
background-repeat: no-repeat;
background-attachment: fixed;
}
::scrollbar {
width: $scrollbar-width;
}
::scrollbar-track {
border-radius: $scrollbar-width;
background-color: rgba(25,147,147,0.1);
}
::scrollbar-thumb {
border-radius: $scrollbar-width;
background-color: $chat-thread-bgd-color;
}
.chat-thread {
margin: 24px auto 0 auto;
padding: 0 20px 0 0;
list-style: none;
overflow-y: scroll;
overflow-x: hidden;
}
.chat-thread li {
position: relative;
clear: both;
display: inline-block;
padding: 16px 40px 16px 20px;
margin: 0 0 20px 0;
font: 16px/20px 'Noto Sans', sans-serif;
border-radius: 10px;
background-color: $chat-thread-bgd-color;
}
/* Chat - Avatar */
.chat-thread li:before {
position: absolute;
top: 0;
width: $chat-thread-avatar-size;
height: $chat-thread-avatar-size;
border-radius: $chat-thread-avatar-size;
content: '';
}
/* Chat - Speech Bubble Arrow */
.chat-thread li:after {
position: absolute;
top: $chat-thread-msg-arrow-size;
content: '';
width: 0;
height: 0;
border-top: $chat-thread-msg-arrow-size solid $chat-thread-bgd-color;
}
.chat-thread li:nth-child(odd) {
animation: show-chat-odd 0.15s 1 ease-in;
animation: show-chat-odd 0.15s 1 ease-in;
animation: show-chat-odd 0.15s 1 ease-in;
float: right;
margin-right: $chat-thread-offset;
color: #0AD5C1;
}
.chat-thread li:nth-child(odd):before {
right: -$chat-thread-offset;
// Placeholder avatar 1
background-image: url();
}
.chat-thread li:nth-child(odd):after {
border-right: $chat-thread-msg-arrow-size solid transparent;
right: -$chat-thread-msg-arrow-size;
}
.chat-thread li:nth-child(even) {
animation: show-chat-even 0.15s 1 ease-in;
animation: show-chat-even 0.15s 1 ease-in;
animation: show-chat-even 0.15s 1 ease-in;
float: left;
margin-left: $chat-thread-offset;
color: #0EC879;
}
.chat-thread li:nth-child(even):before {
left: -$chat-thread-offset;
// Placeholder avatar 2
background-image: url();
}
.chat-thread li:nth-child(even):after {
border-left: $chat-thread-msg-arrow-size solid transparent;
left: -$chat-thread-msg-arrow-size;
}
.chat-window {
position: fixed;
bottom: 18px;
}
.chat-window-message {
width: 100%;
height: 48px;
font: 32px/48px 'Noto Sans', sans-serif;
background: none;
color: #0AD5C1;
border: 0;
border-bottom: 1px solid $chat-thread-bgd-color;
outline: none;
}
// A tiny bit responsive...
// --------------------------------------
/* Small screens */
@media all and (max-width: 767px) {
.chat-thread {
width: 90%;
height: 260px;
}
.chat-window {
left: 5%;
width: 90%;
}
}
/* Medium and large screens */
@media all and (min-width: 768px) {
.chat-thread {
width: 50%;
height: 320px;
}
.chat-window {
left: 25%;
width: 50%;
}
}
// Animation
// --------------------------------------
@keyframes show-chat-even {
0% {
margin-left: -480px;
}
100% {
margin-left: 0;
}
}
@-moz-keyframes show-chat-even {
0% {
margin-left: -480px;
}
100% {
margin-left: 0;
}
}
@-webkit-keyframes show-chat-even {
0% {
margin-left: -480px;
}
100% {
margin-left: 0;
}
}
@keyframes show-chat-odd {
0% {
margin-right: -480px;
}
100% {
margin-right: 0;
}
}
@-moz-keyframes show-chat-odd {
0% {
margin-right: -480px;
}
100% {
margin-right: 0;
}
}
@-webkit-keyframes show-chat-odd {
0% {
margin-right: -480px;
}
100% {
margin-right: 0;
}
}
View Compiled
// Chat (WebRTC)
//
// Currently supported in Chrome and Firefox only.
// WebRTC support is ultra basic at the moment - send/receive // in current window only.
// Design based on Bookmarks app by // Eyal Zuri - http://dribbble.com/shots/1261465-Bookmarks-app-gif
//
// The below JS has been adapted from this excellent RTCDataChannel demo
// http://simpl.info/rtcdatachannel/
var sendChannel,
receiveChannel,
chatWindow = document.querySelector('.chat-window'),
chatWindowMessage = document.querySelector('.chat-window-message'),
chatThread = document.querySelector('.chat-thread');
// Create WebRTC connection
createConnection();
// On form submit, send message
chatWindow.onsubmit = function (e) {
e.preventDefault();
sendData();
return false;
};
function createConnection () {
var servers = null;
if (window.mozRTCPeerConnection) {
window.localPeerConnection = new mozRTCPeerConnection(servers, {
optional: [{
RtpDataChannels: true
}]
});
} else {
window.localPeerConnection = new webkitRTCPeerConnection(servers, {
optional: [{
RtpDataChannels: true
}]
});
}
try {
// Reliable Data Channels not yet supported in Chrome
sendChannel = localPeerConnection.createDataChannel('sendDataChannel', {
reliable: false
});
} catch (e) {
}
localPeerConnection.onicecandidate = gotLocalCandidate;
sendChannel.onopen = handleSendChannelStateChange;
sendChannel.onclose = handleSendChannelStateChange;
if (window.mozRTCPeerConnection) {
window.remotePeerConnection = new mozRTCPeerConnection(servers, {
optional: [{
RtpDataChannels: true
}]
});
} else {
window.remotePeerConnection = new webkitRTCPeerConnection(servers, {
optional: [{
RtpDataChannels: true
}]
});
}
remotePeerConnection.onicecandidate = gotRemoteIceCandidate;
remotePeerConnection.ondatachannel = gotReceiveChannel;
// Firefox seems to require an error callback
localPeerConnection.createOffer(gotLocalDescription, function (err) {
});
}
function sendData () {
sendChannel.send(chatWindowMessage.value);
}
function gotLocalDescription (desc) {
localPeerConnection.setLocalDescription(desc);
remotePeerConnection.setRemoteDescription(desc);
// Firefox seems to require an error callback
remotePeerConnection.createAnswer(gotRemoteDescription, function (err) {
});
}
function gotRemoteDescription (desc) {
remotePeerConnection.setLocalDescription(desc);
localPeerConnection.setRemoteDescription(desc);
}
function gotLocalCandidate (event) {
if (event.candidate) {
remotePeerConnection.addIceCandidate(event.candidate);
}
}
function gotRemoteIceCandidate (event) {
if (event.candidate) {
localPeerConnection.addIceCandidate(event.candidate);
}
}
function gotReceiveChannel (event) {
receiveChannel = event.channel;
receiveChannel.onmessage = handleMessage;
receiveChannel.onopen = handleReceiveChannelStateChange;
receiveChannel.onclose = handleReceiveChannelStateChange;
}
function handleMessage (event) {
var chatNewThread = document.createElement('li'),
chatNewMessage = document.createTextNode(event.data);
// Add message to chat thread and scroll to bottom
chatNewThread.appendChild(chatNewMessage);
chatThread.appendChild(chatNewThread);
chatThread.scrollTop = chatThread.scrollHeight;
// Clear text value
chatWindowMessage.value = '';
}
function handleSendChannelStateChange () {
var readyState = sendChannel.readyState;
if (readyState == 'open') {
chatWindowMessage.disabled = false;
chatWindowMessage.focus();
chatWindowMessage.placeholder = "";
} else {
chatWindowMessage.disabled = true;
}
}
function handleReceiveChannelStateChange () {
var readyState = receiveChannel.readyState;
}
This Pen doesn't use any external CSS resources.