2011-09-28 09:16:26 -04:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2009-09-22 15:24:14 -04:00
|
|
|
|
2012-02-15 16:16:23 +01:00
|
|
|
const Clutter = imports.gi.Clutter;
|
2009-09-22 15:24:14 -04:00
|
|
|
const Lang = imports.lang;
|
2010-04-09 16:43:27 -04:00
|
|
|
const Meta = imports.gi.Meta;
|
2013-01-31 16:07:16 +01:00
|
|
|
const Signals = imports.signals;
|
2010-02-26 23:13:11 +03:00
|
|
|
const St = imports.gi.St;
|
2012-12-15 02:41:03 +01:00
|
|
|
const Shell = imports.gi.Shell;
|
2009-09-22 15:24:14 -04:00
|
|
|
|
2010-03-17 15:36:57 +01:00
|
|
|
const Params = imports.misc.params;
|
|
|
|
const Tweener = imports.ui.tweener;
|
|
|
|
|
2012-02-06 17:28:48 -05:00
|
|
|
const DEFAULT_FADE_FACTOR = 0.4;
|
|
|
|
|
2012-12-15 02:41:03 +01:00
|
|
|
const GLSL_DIM_EFFECT_DECLARATIONS = '\
|
|
|
|
float compute_dim_factor (const vec2 coords) {\
|
|
|
|
vec2 dist = coords - vec2(0.5, 0.5); \
|
|
|
|
float elipse_radius = 0.5; \
|
|
|
|
/* interpolate darkening value, based on distance from screen center */ \
|
|
|
|
float val = min(length(dist), elipse_radius); \
|
|
|
|
return mix(0.3, 1.0, val / elipse_radius) * 0.4; \
|
|
|
|
}';
|
|
|
|
const GLSL_DIM_EFFECT_CODE = '\
|
|
|
|
float a = compute_dim_factor (cogl_tex_coord0_in.xy);\
|
|
|
|
cogl_color_out = vec4(0, 0, 0, cogl_color_in.a * a);'
|
|
|
|
;
|
|
|
|
|
|
|
|
const RadialShaderQuad = new Lang.Class({
|
|
|
|
Name: 'RadialShaderQuad',
|
|
|
|
Extends: Shell.GLSLQuad,
|
|
|
|
|
|
|
|
vfunc_build_pipeline: function() {
|
|
|
|
this.add_glsl_snippet(Shell.SnippetHook.FRAGMENT,
|
|
|
|
GLSL_DIM_EFFECT_DECLARATIONS,
|
|
|
|
GLSL_DIM_EFFECT_CODE,
|
|
|
|
true);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2009-09-22 15:24:14 -04:00
|
|
|
/**
|
|
|
|
* Lightbox:
|
|
|
|
* @container: parent Clutter.Container
|
2010-03-17 15:36:57 +01:00
|
|
|
* @params: (optional) additional parameters:
|
|
|
|
* - inhibitEvents: whether to inhibit events for @container
|
|
|
|
* - width: shade actor width
|
|
|
|
* - height: shade actor height
|
2012-02-06 17:28:48 -05:00
|
|
|
* - fadeInTime: seconds used to fade in
|
|
|
|
* - fadeOutTime: seconds used to fade out
|
2009-09-22 15:24:14 -04:00
|
|
|
*
|
|
|
|
* Lightbox creates a dark translucent "shade" actor to hide the
|
|
|
|
* contents of @container, and allows you to specify particular actors
|
|
|
|
* in @container to highlight by bringing them above the shade. It
|
|
|
|
* tracks added and removed actors in @container while the lightboxing
|
|
|
|
* is active, and ensures that all actors are returned to their
|
|
|
|
* original stacking order when the lightboxing is removed. (However,
|
|
|
|
* if actors are restacked by outside code while the lightboxing is
|
|
|
|
* active, the lightbox may later revert them back to their original
|
|
|
|
* order.)
|
|
|
|
*
|
|
|
|
* By default, the shade window will have the height and width of
|
|
|
|
* @container and will track any changes in its size. You can override
|
2010-03-17 15:36:57 +01:00
|
|
|
* this by passing an explicit width and height in @params.
|
2009-09-22 15:24:14 -04:00
|
|
|
*/
|
2011-11-20 18:56:27 +01:00
|
|
|
const Lightbox = new Lang.Class({
|
|
|
|
Name: 'Lightbox',
|
2009-09-22 15:24:14 -04:00
|
|
|
|
2010-03-17 15:36:57 +01:00
|
|
|
_init : function(container, params) {
|
|
|
|
params = Params.parse(params, { inhibitEvents: false,
|
|
|
|
width: null,
|
|
|
|
height: null,
|
2013-05-12 17:39:44 +02:00
|
|
|
fadeFactor: DEFAULT_FADE_FACTOR,
|
2012-12-15 02:41:03 +01:00
|
|
|
radialEffect: false,
|
2010-03-17 15:36:57 +01:00
|
|
|
});
|
|
|
|
|
2009-09-22 15:24:14 -04:00
|
|
|
this._container = container;
|
|
|
|
this._children = container.get_children();
|
2012-02-06 17:28:48 -05:00
|
|
|
this._fadeFactor = params.fadeFactor;
|
2012-12-15 02:41:03 +01:00
|
|
|
if (params.radialEffect)
|
|
|
|
this.actor = new RadialShaderQuad({ x: 0,
|
|
|
|
y: 0,
|
|
|
|
reactive: params.inhibitEvents });
|
|
|
|
else
|
|
|
|
this.actor = new St.Bin({ x: 0,
|
|
|
|
y: 0,
|
|
|
|
style_class: 'lightbox',
|
|
|
|
reactive: params.inhibitEvents });
|
2009-09-22 15:24:14 -04:00
|
|
|
|
|
|
|
container.add_actor(this.actor);
|
|
|
|
this.actor.raise_top();
|
2010-03-17 15:36:57 +01:00
|
|
|
this.actor.hide();
|
2012-02-06 17:28:48 -05:00
|
|
|
this.shown = false;
|
2009-09-22 15:24:14 -04:00
|
|
|
|
2010-02-18 16:43:58 +01:00
|
|
|
this.actor.connect('destroy', Lang.bind(this, this._onDestroy));
|
2009-09-22 15:24:14 -04:00
|
|
|
|
2010-03-17 15:36:57 +01:00
|
|
|
if (params.width && params.height) {
|
|
|
|
this.actor.width = params.width;
|
|
|
|
this.actor.height = params.height;
|
2009-09-22 15:24:14 -04:00
|
|
|
} else {
|
2012-02-15 16:16:23 +01:00
|
|
|
let constraint = new Clutter.BindConstraint({ source: container,
|
|
|
|
coordinate: Clutter.BindCoordinate.ALL });
|
|
|
|
this.actor.add_constraint(constraint);
|
2009-09-22 15:24:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
this._actorAddedSignalId = container.connect('actor-added', Lang.bind(this, this._actorAdded));
|
|
|
|
this._actorRemovedSignalId = container.connect('actor-removed', Lang.bind(this, this._actorRemoved));
|
|
|
|
|
|
|
|
this._highlighted = null;
|
|
|
|
},
|
|
|
|
|
|
|
|
_actorAdded : function(container, newChild) {
|
|
|
|
let children = this._container.get_children();
|
|
|
|
let myIndex = children.indexOf(this.actor);
|
|
|
|
let newChildIndex = children.indexOf(newChild);
|
|
|
|
|
2009-09-22 17:48:44 -04:00
|
|
|
if (newChildIndex > myIndex) {
|
2009-09-22 15:24:14 -04:00
|
|
|
// The child was added above the shade (presumably it was
|
|
|
|
// made the new top-most child). Move it below the shade,
|
|
|
|
// and add it to this._children as the new topmost actor.
|
|
|
|
newChild.lower(this.actor);
|
|
|
|
this._children.push(newChild);
|
2009-09-22 17:48:44 -04:00
|
|
|
} else if (newChildIndex == 0) {
|
|
|
|
// Bottom of stack
|
|
|
|
this._children.unshift(newChild);
|
2009-09-22 15:24:14 -04:00
|
|
|
} else {
|
|
|
|
// Somewhere else; insert it into the correct spot
|
2009-09-22 17:48:44 -04:00
|
|
|
let prevChild = this._children.indexOf(children[newChildIndex - 1]);
|
|
|
|
if (prevChild != -1) // paranoia
|
|
|
|
this._children.splice(prevChild + 1, 0, newChild);
|
2009-09-22 15:24:14 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2013-05-12 17:39:44 +02:00
|
|
|
show: function(fadeInTime) {
|
|
|
|
fadeInTime = fadeInTime || 0;
|
|
|
|
|
2013-01-17 23:55:17 +01:00
|
|
|
Tweener.removeTweens(this.actor);
|
2013-05-12 17:39:44 +02:00
|
|
|
if (fadeInTime != 0) {
|
2012-02-06 17:28:48 -05:00
|
|
|
this.shown = false;
|
2010-03-17 15:36:57 +01:00
|
|
|
this.actor.opacity = 0;
|
|
|
|
Tweener.addTween(this.actor,
|
2012-02-06 17:28:48 -05:00
|
|
|
{ opacity: 255 * this._fadeFactor,
|
2013-05-12 17:39:44 +02:00
|
|
|
time: fadeInTime,
|
2012-02-06 17:28:48 -05:00
|
|
|
transition: 'easeOutQuad',
|
|
|
|
onComplete: Lang.bind(this, function() {
|
|
|
|
this.shown = true;
|
2013-01-17 23:55:17 +01:00
|
|
|
this.emit('shown');
|
2012-02-06 17:28:48 -05:00
|
|
|
})
|
2010-03-17 15:36:57 +01:00
|
|
|
});
|
|
|
|
} else {
|
2012-02-06 17:28:48 -05:00
|
|
|
this.actor.opacity = 255 * this._fadeFactor;
|
|
|
|
this.shown = true;
|
2013-01-17 23:55:17 +01:00
|
|
|
this.emit('shown');
|
2010-03-17 15:36:57 +01:00
|
|
|
}
|
|
|
|
this.actor.show();
|
|
|
|
},
|
|
|
|
|
2013-05-12 17:39:44 +02:00
|
|
|
hide: function(fadeOutTime) {
|
|
|
|
fadeOutTime = fadeOutTime || 0;
|
|
|
|
|
2012-02-06 17:28:48 -05:00
|
|
|
this.shown = false;
|
2013-01-17 23:55:17 +01:00
|
|
|
Tweener.removeTweens(this.actor);
|
2013-05-12 17:39:44 +02:00
|
|
|
if (fadeOutTime != 0) {
|
2010-03-17 15:36:57 +01:00
|
|
|
Tweener.addTween(this.actor,
|
|
|
|
{ opacity: 0,
|
2013-05-12 17:39:44 +02:00
|
|
|
time: fadeOutTime,
|
2010-03-17 15:36:57 +01:00
|
|
|
transition: 'easeOutQuad',
|
|
|
|
onComplete: Lang.bind(this, function() {
|
|
|
|
this.actor.hide();
|
|
|
|
})
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.actor.hide();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2009-09-22 15:24:14 -04:00
|
|
|
_actorRemoved : function(container, child) {
|
|
|
|
let index = this._children.indexOf(child);
|
|
|
|
if (index != -1) // paranoia
|
|
|
|
this._children.splice(index, 1);
|
|
|
|
|
|
|
|
if (child == this._highlighted)
|
|
|
|
this._highlighted = null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* highlight:
|
|
|
|
* @window: actor to highlight
|
|
|
|
*
|
|
|
|
* Highlights the indicated actor and unhighlights any other
|
|
|
|
* currently-highlighted actor. With no arguments or a false/null
|
|
|
|
* argument, all actors will be unhighlighted.
|
|
|
|
*/
|
|
|
|
highlight : function(window) {
|
|
|
|
if (this._highlighted == window)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Walk this._children raising and lowering actors as needed.
|
|
|
|
// Things get a little tricky if the to-be-raised and
|
|
|
|
// to-be-lowered actors were originally adjacent, in which
|
|
|
|
// case we may need to indicate some *other* actor as the new
|
|
|
|
// sibling of the to-be-lowered one.
|
|
|
|
|
|
|
|
let below = this.actor;
|
2009-09-22 17:48:44 -04:00
|
|
|
for (let i = this._children.length - 1; i >= 0; i--) {
|
2009-09-22 15:24:14 -04:00
|
|
|
if (this._children[i] == window)
|
|
|
|
this._children[i].raise_top();
|
|
|
|
else if (this._children[i] == this._highlighted)
|
|
|
|
this._children[i].lower(below);
|
|
|
|
else
|
|
|
|
below = this._children[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
this._highlighted = window;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* destroy:
|
|
|
|
*
|
2010-02-18 16:43:58 +01:00
|
|
|
* Destroys the lightbox.
|
2009-09-22 15:24:14 -04:00
|
|
|
*/
|
|
|
|
destroy : function() {
|
2010-02-18 16:43:58 +01:00
|
|
|
this.actor.destroy();
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* _onDestroy:
|
|
|
|
*
|
|
|
|
* This is called when the lightbox' actor is destroyed, either
|
|
|
|
* by destroying its container or by explicitly calling this.destroy().
|
|
|
|
*/
|
|
|
|
_onDestroy: function() {
|
2009-09-22 15:24:14 -04:00
|
|
|
this._container.disconnect(this._actorAddedSignalId);
|
|
|
|
this._container.disconnect(this._actorRemovedSignalId);
|
|
|
|
|
|
|
|
this.highlight(null);
|
|
|
|
}
|
2011-11-20 18:56:27 +01:00
|
|
|
});
|
2013-01-17 23:55:17 +01:00
|
|
|
Signals.addSignalMethods(Lightbox.prototype);
|