mirror of
https://github.com/discourse/discourse.git
synced 2025-02-19 16:12:45 +08:00
153 lines
4.6 KiB
JavaScript
153 lines
4.6 KiB
JavaScript
/**
|
|
This view handles rendering of the header of the site
|
|
|
|
@class HeaderView
|
|
@extends Discourse.View
|
|
@namespace Discourse
|
|
@module Discourse
|
|
**/
|
|
Discourse.HeaderView = Discourse.View.extend({
|
|
tagName: 'header',
|
|
classNames: ['d-header', 'clearfix'],
|
|
classNameBindings: ['editingTopic'],
|
|
templateName: 'header',
|
|
siteBinding: 'Discourse.site',
|
|
currentUserBinding: 'Discourse.currentUser',
|
|
categoriesBinding: 'site.categories',
|
|
topicBinding: 'Discourse.router.topicController.content',
|
|
|
|
showDropdown: function($target) {
|
|
var elementId = $target.data('dropdown') || $target.data('notifications'),
|
|
$dropdown = $("#" + elementId),
|
|
$li = $target.closest('li'),
|
|
$ul = $target.closest('ul'),
|
|
$html = $('html');
|
|
|
|
var hideDropdown = function() {
|
|
$dropdown.fadeOut('fast');
|
|
$li.removeClass('active');
|
|
$html.data('hide-dropdown', null);
|
|
return $html.off('click.d-dropdown');
|
|
};
|
|
|
|
// if a dropdown is active and the user clics on it, close it
|
|
if($li.hasClass('active')) { return hideDropdown(); }
|
|
// otherwhise, mark it as active
|
|
$li.addClass('active');
|
|
// hide the other dropdowns
|
|
$('li', $ul).not($li).removeClass('active');
|
|
$('.d-dropdown').not($dropdown).fadeOut('fast');
|
|
// fade it fast
|
|
$dropdown.fadeIn('fast');
|
|
// autofocus any text input field
|
|
$dropdown.find('input[type=text]').focus().select();
|
|
|
|
$html.on('click.d-dropdown', function(e) {
|
|
return $(e.target).closest('.d-dropdown').length > 0 ? true : hideDropdown();
|
|
});
|
|
|
|
$html.data('hide-dropdown', hideDropdown);
|
|
|
|
return false;
|
|
},
|
|
|
|
showNotifications: function() {
|
|
|
|
var headerView = this;
|
|
Discourse.ajax('/notifications').then(function(result) {
|
|
headerView.set('notifications', result.map(function(n) {
|
|
return Discourse.Notification.create(n);
|
|
}));
|
|
|
|
// We've seen all the notifications now
|
|
headerView.set('currentUser.unread_notifications', 0);
|
|
headerView.showDropdown($('#user-notifications'));
|
|
});
|
|
return false;
|
|
},
|
|
|
|
examineDockHeader: function() {
|
|
var $body, offset, outlet;
|
|
if (!this.docAt) {
|
|
outlet = $('#main-outlet');
|
|
if (!(outlet && outlet.length === 1)) return;
|
|
this.docAt = outlet.offset().top;
|
|
}
|
|
offset = window.pageYOffset || $('html').scrollTop();
|
|
if (offset >= this.docAt) {
|
|
if (!this.dockedHeader) {
|
|
$body = $('body');
|
|
$body.addClass('docked');
|
|
this.dockedHeader = true;
|
|
}
|
|
} else {
|
|
if (this.dockedHeader) {
|
|
$('body').removeClass('docked');
|
|
this.dockedHeader = false;
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
Display the correct logo in the header, showing a custom small icon if it exists.
|
|
|
|
@property logoHTML
|
|
**/
|
|
logoHTML: function() {
|
|
var result = "<div class='title'><a href='" + Discourse.getURL("/") + "'>";
|
|
if (this.get('controller.showExtraInfo')) {
|
|
var logo = Discourse.SiteSettings.logo_small_url;
|
|
if (logo && logo.length > 1) {
|
|
result += "<img class='logo-small' src='" + logo + "' width='33' height='33'>";
|
|
} else {
|
|
result += "<i class='icon-home'></i>";
|
|
}
|
|
} else {
|
|
result += "<img class='logo-big' src=\"" + Discourse.SiteSettings.logo_url + "\" alt=\"" + Discourse.SiteSettings.title + "\" id='site-logo'>";
|
|
}
|
|
result += "</a></div>";
|
|
return new Handlebars.SafeString(result);
|
|
}.property('controller.showExtraInfo'),
|
|
|
|
willDestroyElement: function() {
|
|
$(window).unbind('scroll.discourse-dock');
|
|
return $(document).unbind('touchmove.discourse-dock');
|
|
},
|
|
|
|
didInsertElement: function() {
|
|
var _this = this;
|
|
this.$('a[data-dropdown]').on('click', function(e) {
|
|
return _this.showDropdown($(e.currentTarget));
|
|
});
|
|
this.$('a.unread-private-messages, a.unread-notifications, a[data-notifications]').on('click', function(e) {
|
|
return _this.showNotifications(e);
|
|
});
|
|
$(window).bind('scroll.discourse-dock', function() {
|
|
return _this.examineDockHeader();
|
|
});
|
|
$(document).bind('touchmove.discourse-dock', function() {
|
|
return _this.examineDockHeader();
|
|
});
|
|
this.examineDockHeader();
|
|
|
|
// Delegate ESC to the composer
|
|
return $('body').on('keydown.header', function(e) {
|
|
// Hide dropdowns
|
|
if (e.which === 27) {
|
|
_this.$('li').removeClass('active');
|
|
_this.$('.d-dropdown').fadeOut('fast');
|
|
}
|
|
if (_this.get('editingTopic')) {
|
|
if (e.which === 13) {
|
|
_this.finishedEdit();
|
|
}
|
|
if (e.which === 27) {
|
|
return _this.cancelEdit();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
|