Support for scrolling to another post in the iframe; more complicated than you'd think!

This commit is contained in:
Robin Ward 2014-01-03 14:45:22 -05:00
parent 2a79ed97ed
commit c762e3c4b1
3 changed files with 62 additions and 6 deletions

View File

@ -12,10 +12,10 @@
<%- if @topic_view.posts.present? %> <%- if @topic_view.posts.present? %>
<%- @topic_view.posts.each do |post| %> <%- @topic_view.posts.each do |post| %>
<article class='post<%- if post.trashed? %> deleted<% end %>'> <article class='post<%- if post.trashed? %> deleted<% end %>' id='post-<%= post.id.to_s %>'>
<%= link_to post.created_at.strftime("%e %b %Y"), post.url, class: 'post-date', target: "_blank" %> <%= link_to post.created_at.strftime("%e %b %Y"), post.url, class: 'post-date', target: "_blank" %>
<%- if post.reply_to_post.present? %> <%- if post.reply_to_post.present? %>
<%= link_to I18n.t('embed.in_reply_to', username: post.reply_to_post.username), post.reply_to_post.url, class: 'in-reply-to', target: "_blank" %> <%= link_to I18n.t('embed.in_reply_to', username: post.reply_to_post.username), post.reply_to_post.url, 'data-link-to-post' => post.reply_to_post.id.to_s, :class => 'in-reply-to' %>
<%- end %> <%- end %>
<div class='author'> <div class='author'>
@ -26,8 +26,12 @@
<%= raw post.cooked %> <%= raw post.cooked %>
<%- if post.reply_count > 0 %> <%- if post.reply_count > 0 %>
<%- if post.reply_count == 1 %>
<%= link_to I18n.t('embed.replies', count: post.reply_count), post.url, 'data-link-to-post' => post.replies.first.id.to_s, :class => 'post-replies' %>
<% else %>
<%= link_to I18n.t('embed.replies', count: post.reply_count), post.url, class: 'post-replies', target: "_blank" %> <%= link_to I18n.t('embed.replies', count: post.reply_count), post.url, class: 'post-replies', target: "_blank" %>
<%- end %> <%- end %>
<%- end %>
</div> </div>
<div style='clear: both'></div> <div style='clear: both'></div>

View File

@ -5,12 +5,38 @@
<script> <script>
(function() { (function() {
window.onload = function() {
function postUp(msg) {
if (parent) { if (parent) {
parent.postMessage(msg, '<%= request.referer %>');
}
}
function clickPostLink(e) {
var postId = e.target.getAttribute('data-link-to-post');
if (postId) {
var postElement = document.getElementById('post-' + postId);
if (postElement) {
var rect = postElement.getBoundingClientRect();
if (rect && rect.top) {
postUp({type: 'discourse-scroll', top: rect.top});
e.preventDefault();
return false;
}
}
}
}
window.onload = function() {
// Send a post message with our loaded height // Send a post message with our loaded height
parent.postMessage({type: 'discourse-resize', height: document['body'].offsetHeight}, '<%= request.referer %>'); postUp({type: 'discourse-resize', height: document['body'].offsetHeight});
}
var postLinks = document.querySelectorAll("a[data-link-to-post]");
for (var i=0; i<postLinks.length; i++) {
postLinks[i].onclick = clickPostLink;
} }
};
})(); })();
</script> </script>
</head> </head>

View File

@ -11,6 +11,26 @@
iframe.scrolling = "no"; iframe.scrolling = "no";
comments.appendChild(iframe); comments.appendChild(iframe);
// Thanks http://amendsoft-javascript.blogspot.ca/2010/04/find-x-and-y-coordinate-of-html-control.html
function findPosY(obj)
{
var top = 0;
if(obj.offsetParent)
{
while(1)
{
top += obj.offsetTop;
if(!obj.offsetParent)
break;
obj = obj.offsetParent;
}
}
else if(obj.y)
{
top += obj.y;
}
return top;
}
function postMessageReceived(e) { function postMessageReceived(e) {
if (!e) { return; } if (!e) { return; }
@ -20,6 +40,12 @@
if (e.data.type === 'discourse-resize' && e.data.height) { if (e.data.type === 'discourse-resize' && e.data.height) {
iframe.height = e.data.height + "px"; iframe.height = e.data.height + "px";
} }
if (e.data.type === 'discourse-scroll' && e.data.top) {
// find iframe offset
var destY = findPosY(iframe) + e.data.top;
window.scrollTo(0, destY);
}
} }
} }
window.addEventListener('message', postMessageReceived, false); window.addEventListener('message', postMessageReceived, false);