windowManager: fix up accounting of dimmed windows

Simplify the accounting of which windows we should dim by checking
the current state of windows rather than trying to track changes,
and by keeping a list of dimmed windows rather than a list of windows
with a dimmed parent. Remove windows from the list of dimmed windows
when they are destroyed.

This should fix problems where destroyed windows could end up in
the list of dimmed windows.

https://bugzilla.gnome.org/show_bug.cgi?id=644167
This commit is contained in:
Owen W. Taylor 2011-03-07 22:09:41 -05:00
parent 2ea762cfc9
commit c6170ed751

View File

@ -123,11 +123,11 @@ WindowManager.prototype = {
Main.overview.connect('showing', Lang.bind(this, function() { Main.overview.connect('showing', Lang.bind(this, function() {
for (let i = 0; i < this._dimmedWindows.length; i++) for (let i = 0; i < this._dimmedWindows.length; i++)
this._undimParentWindow(this._dimmedWindows[i], true); this._undimWindow(this._dimmedWindows[i], true);
})); }));
Main.overview.connect('hiding', Lang.bind(this, function() { Main.overview.connect('hiding', Lang.bind(this, function() {
for (let i = 0; i < this._dimmedWindows.length; i++) for (let i = 0; i < this._dimmedWindows.length; i++)
this._dimParentWindow(this._dimmedWindows[i], true); this._dimWindow(this._dimmedWindows[i], true);
})); }));
}, },
@ -234,41 +234,39 @@ WindowManager.prototype = {
_unmaximizeWindowDone : function(shellwm, actor) { _unmaximizeWindowDone : function(shellwm, actor) {
}, },
_parentHasOtherAttachedDialog: function(parent, self) { _hasAttachedDialogs: function(window, ignoreWindow) {
var count = 0; var count = 0;
parent.foreach_transient(function(win) { window.foreach_transient(function(win) {
if (win.get_window_type() == Meta.WindowType.MODAL_DIALOG && win != self) if (win != ignoreWindow && win.get_window_type() == Meta.WindowType.MODAL_DIALOG)
count++; count++;
return false; return false;
}); });
return count != 0; return count != 0;
}, },
_markParentWindowAsDimmable: function(actor, animate) { _checkDimming: function(window, ignoreWindow) {
if (Meta.prefs_get_attach_modal_dialogs()) { let shouldDim = Meta.prefs_get_attach_modal_dialogs() && this._hasAttachedDialogs(window, ignoreWindow);
this._dimmedWindows.push(actor);
if (this._shouldAnimate()) if (shouldDim && !window._dimmed) {
this._dimParentWindow(actor, animate); window._dimmed = true;
this._dimmedWindows.push(window);
if (!Main.overview.visible)
this._dimWindow(window, true);
} else if (!shouldDim && window._dimmed) {
window._dimmed = false;
this._dimmedWindows = this._dimmedWindows.filter(function(win) {
return win != window;
});
if (!Main.overview.visible)
this._undimWindow(window, true);
} }
}, },
_unmarkParentWindowAsDimmable: function(actor, animate) { _dimWindow: function(window, animate) {
if (!Main.overview.visible) let actor = window.get_compositor_private();
this._undimParentWindow(actor, true); if (!actor)
this._dimmedWindows = this._dimmedWindows.filter(function(win) {
return win != actor;
});
},
_dimParentWindow: function(actor, animate) {
let meta = actor.get_meta_window();
let parent = meta.get_transient_for();
if (!parent)
return; return;
let parentActor = parent.get_compositor_private(); let texture = actor.get_texture();
if (!parentActor || this._parentHasOtherAttachedDialog(parent, meta))
return;
let texture = parentActor.get_texture();
if (animate) if (animate)
Tweener.addTween(getWindowDimmer(texture), Tweener.addTween(getWindowDimmer(texture),
{ dimFraction: 1.0, { dimFraction: 1.0,
@ -279,15 +277,11 @@ WindowManager.prototype = {
getWindowDimmer(texture).dimFraction = 1.0; getWindowDimmer(texture).dimFraction = 1.0;
}, },
_undimParentWindow: function(actor, animate) { _undimWindow: function(window, animate) {
let meta = actor.get_meta_window(); let actor = window.get_compositor_private();
let parent = meta.get_transient_for(); if (!actor)
if (!parent)
return; return;
let parentActor = parent.get_compositor_private(); let texture = actor.get_texture();
if (!parentActor || this._parentHasOtherAttachedDialog(parent, meta))
return;
let texture = parentActor.get_texture();
if (animate) if (animate)
Tweener.addTween(getWindowDimmer(texture), Tweener.addTween(getWindowDimmer(texture),
{ dimFraction: 0.0, { dimFraction: 0.0,
@ -304,17 +298,19 @@ WindowManager.prototype = {
let type = actor.meta_window.get_window_type(); let type = actor.meta_window.get_window_type();
if (type == actor._windowType) if (type == actor._windowType)
return; return;
if (type == Meta.WindowType.MODAL_DIALOG) if (type == Meta.WindowType.MODAL_DIALOG ||
this._markParentWindowAsDimmable(actor, true); actor._windowType == Meta.WindowType.MODAL_DIALOG) {
else if (actor._windowType == Meta.WindowType.MODAL_DIALOG) let parent = actor.get_meta_window().get_transient_for();
this._unmarkParentWindowAsDimmable(actor, true); if (parent)
this._checkDimming(parent);
}
actor._windowType = type; actor._windowType = type;
})); }));
if (actor.meta_window.get_window_type() == Meta.WindowType.MODAL_DIALOG if (actor.meta_window.get_window_type() == Meta.WindowType.MODAL_DIALOG
&& Meta.prefs_get_attach_modal_dialogs() && Meta.prefs_get_attach_modal_dialogs()
&& actor.get_meta_window().get_transient_for()) { && actor.get_meta_window().get_transient_for()) {
this._markParentWindowAsDimmable(actor, true); this._checkDimming(actor.get_meta_window().get_transient_for());
if (this._shouldAnimate()) { if (this._shouldAnimate()) {
actor.set_scale(1.0, 0.0); actor.set_scale(1.0, 0.0);
actor.show(); actor.show();
@ -374,14 +370,20 @@ WindowManager.prototype = {
}, },
_destroyWindow : function(shellwm, actor) { _destroyWindow : function(shellwm, actor) {
let parent = actor.meta_window.get_transient_for(); let window = actor.meta_window;
let parent = window.get_transient_for();
if (actor._notifyWindowTypeSignalId) { if (actor._notifyWindowTypeSignalId) {
actor.meta_window.disconnect(actor._notifyWindowTypeSignalId); window.disconnect(actor._notifyWindowTypeSignalId);
actor._notifyWindowTypeSignalId = 0; actor._notifyWindowTypeSignalId = 0;
} }
while (actor.meta_window.get_window_type() == Meta.WindowType.MODAL_DIALOG if (window._dimmed) {
this._dimmedWindows = this._dimmedWindows.filter(function(win) {
return win != window;
});
}
while (window.get_window_type() == Meta.WindowType.MODAL_DIALOG
&& parent) { && parent) {
this._unmarkParentWindowAsDimmable(actor, true); this._checkDimming(parent, window);
if (!Meta.prefs_get_attach_modal_dialogs() if (!Meta.prefs_get_attach_modal_dialogs()
|| !this._shouldAnimate()) || !this._shouldAnimate())
break; break;