notice-manager/js/notice_manager_panel.js
2022-08-02 06:15:19 +03:00

226 lines
5.5 KiB
JavaScript

/**
* NoticeManager class/module
*
*/
var NoticeManager = (function ($, document) {
let options = window.notice_manager_options;
let selectors_notice = [
"div.notice",
"div.updated",
];
let selectors_warning = [
"div.notice-warning",
"div.update-nag",
];
let selectors_error = [
"div.error",
"div.notice-error",
];
let selectors_all = selectors_notice.concat(selectors_warning, selectors_error);
// wait function used with autoCollapse
let wait = function (ms) {
var dfd = $.Deferred();
setTimeout(dfd.resolve, ms); //callback, timeout till callback
return dfd.promise();
};
let notices;
let button;
let panel;
let haveClosed; // set to true on first close/collect
let dismissNoticesButton;
// bootstrap
// some of these need to run BEFORE document.ready - don't know why really
button = $("#meta-link-notices");
panel = $("#meta-link-notices-wrap");
haveClosed = false;
dismissNoticesButton = $("#meta-link-notices-wrap button.notice-dismiss");
dismissNoticesButton.on("click", () => {
screenMeta.close(panel, button);
if (! haveClosed){
NoticeManager.collectNotices();
}
NoticeManager.addCounter();
});
//original wp focus on click function
button.on("focus.scroll-into-view", (e) => {
if (e.target.scrollIntoView) e.target.scrollIntoView(false);
});
// scroll page to top when closing notice panel
// function used to work with $(this)
// using e.target instead
// not sure if this should perhaps be e.currentTarget
button.on("click", (e) => {
if ($(e.target).hasClass("screen-meta-active")) {
if (haveClosed) {
NoticeManager.addCounter();
}
// $(window).scrollTop(true);
} else {
NoticeManager.removeCounter();
// wait (500).then(function(){ //still jumpy sometimes - but scrolls to correct position 400 ~ 600
// $(window).scrollTop(true);
// });
}
});
/**
* document.on.ready
*/
$(() => {
console.log("NoticeManager.on.ready");
console.log("options");
console.log(options);
// bootstrap notices
// get all notices that are not explicitly marked as `.inline` or `.below-h2`
// we add .update-nag.inline for WordPress Update notice
notices = $( selectors_all.join(', ') )
.not(".inline, .below-h2")
.add("div.update-nag");
/**
* Remove panel if there are no notices on this page
*/
if (options.screen_panel) {
NoticeManager.maybeRemoveNoticesPanel();
}
if (options.screen_panel && options.auto_collect) {
NoticeManager.collectNotices();
} else {
if (options.above_title) {
NoticeManager.moveAboveTitle();
}
}
/**
* auto-open notices panel
*/
if (button.length) {
panel.toggle();
button.addClass("screen-meta-active");
screenMeta.open(panel, button);
}
/**
* auto-close notices panel after short delay
* only auto-close if we have collected notices previously
* only auto-close if no error messages
*/
if (options.auto_collapse) {
wait(4000).then(() => {
if (haveClosed && NoticeManager.getNoticesTopPriority() != 'error') {
screenMeta.close(panel, button);
NoticeManager.addCounter();
}
});
}
}); // end document.on.ready
// prevent jumpy scrollRestoration on reload page
// fixed below on 'beforeunload'
// if (history.scrollRestoration) {
// history.scrollRestoration = 'manual';
//}
/**
* Set history.scrollTop to prevent jump on page refresh when scrollRestoration = auto
*/
$(window).on("beforeunload", () => {
history.pushState(
{ scrollTop: document.body.scrollTop },
document.title,
document.location.pathname
);
});
return {
getNotices: () => notices,
getNoticesTopPriority: () => {
if (notices.filter(":visible").filter(selectors_error.join(", ")).length)
return 'error';
if (notices.filter(":visible").filter(selectors_warning.join(", ")).length)
return 'warning';
return 'notice';
},
/**
* Collect notices into panel.
* Remove dismiss-notices button.
*/
collectNotices: () => {
notices.appendTo(".notice_container").eq(0);
$(".notice_container").removeClass("empty"); // .empty removes padding
haveClosed = true; // initial collection has occurred.
/**
* When dismissible notices are dismissed, check if any notices are left on page.
* If no notices are left - remove Notice Panel entirely
*/
$(document).on(
"DOMNodeRemoved",
"#meta-link-notices-wrap div.notice",
() => {
notices = panel
.find(selectors_all.join(", "))
.filter(":visible");
NoticeManager.maybeRemoveNoticesPanel();
}
);
},
addCounter: () => {
if (!button.children('.plugin-count').length){
button.append(
$("<span/>").text(notices.filter(":visible").length).attr({
class: "plugin-count",
}).addClass(NoticeManager.getNoticesTopPriority())
);
}
},
removeCounter: () => {
button.children(".plugin-count").remove();
},
/**
* Remove meta-links-notices if no notices on page
* Remove screen-meta-links wrapper if no panels on page
*/
maybeRemoveNoticesPanel: () => {
if (!notices.length) {
screenMeta.close(panel, button);
$("#meta-link-notices-link-wrap").detach();
$("#meta-link-notices-wrap").detach();
if (!$("#screen-meta-links").children().length)
$("#screen-meta-links").detach();
}
},
/**
* Move ALL notices above page title.
* Default no-panel action - override WordPress moving notices BELOW title.
* I HATE it when WordPress moves notices below title.
*/
moveAboveTitle: () => {
notices.insertBefore(".wrap:first");
},
};
}(jQuery,document) )