2020-02-06 16:26:37 +01:00
"use strict" ;
2013-02-22 01:25:41 +01:00
/ * *
2020-02-06 16:26:37 +01:00
* EGroupware eTemplate2 - Split panel
*
* @ license http : //opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @ package etemplate
* @ subpackage api
* @ link http : //www.egroupware.org
* @ author Nathan Gray
* @ copyright Nathan Gray 2013
* @ version $Id$
* /
var _ _extends = ( this && this . _ _extends ) || ( function ( ) {
var extendStatics = function ( d , b ) {
extendStatics = Object . setPrototypeOf ||
( { _ _proto _ _ : [ ] } instanceof Array && function ( d , b ) { d . _ _proto _ _ = b ; } ) ||
function ( d , b ) { for ( var p in b ) if ( b . hasOwnProperty ( p ) ) d [ p ] = b [ p ] ; } ;
return extendStatics ( d , b ) ;
} ;
return function ( d , b ) {
extendStatics ( d , b ) ;
function _ _ ( ) { this . constructor = d ; }
d . prototype = b === null ? Object . create ( b ) : ( _ _ . prototype = b . prototype , new _ _ ( ) ) ;
} ;
} ) ( ) ;
Object . defineProperty ( exports , "__esModule" , { value : true } ) ;
2013-02-22 01:25:41 +01:00
/ * e g w : u s e s
2020-02-06 16:26:37 +01:00
/ v e n d o r / b o w e r - a s s e t / j q u e r y / d i s t / j q u e r y . j s ;
jquery . splitter ;
et2 _core _baseWidget ;
2013-02-22 01:25:41 +01:00
* /
2020-02-06 16:26:37 +01:00
var et2 _core _DOMWidget _1 = require ( "./et2_core_DOMWidget" ) ;
var et2 _core _widget _1 = require ( "./et2_core_widget" ) ;
var et2 _core _inheritance _1 = require ( "./et2_core_inheritance" ) ;
2013-02-22 01:25:41 +01:00
/ * *
2020-02-06 16:26:37 +01:00
* A container widget that accepts 2 children , and puts a resize bar between them .
*
* This is the etemplate2 implementation of the traditional split pane / split panel .
* The split can be horizontal or vertical , and can be limited in size . You can also
* turn on double - click docking to minimize one of the children .
*
* @ see http : //methvin.com/splitter/ Uses Splitter
* @ augments et2 _DOMWidget
* /
var et2 _split = /** @class */ ( function ( _super ) {
_ _extends ( et2 _split , _super ) ;
function et2 _split ( _parent , _attrs , _child ) {
var _this =
// Call the inherited constructor
_super . call ( this , _parent , _attrs , et2 _core _inheritance _1 . ClassWithAttributes . extendAttributes ( et2 _split . _attributes , _child || { } ) ) || this ;
_this . div = null ;
_this . dynheight = null ;
_this . div = jQuery ( document . createElement ( "div" ) )
. addClass ( 'et2_split' ) ;
// Create the dynheight component which dynamically scales the inner
// container.
_this . dynheight = new et2 _dynheight ( _this . getParent ( ) . getDOMNode ( ) || _this . getInstanceManager ( ) . DOMContainer , _this . div , 100 ) ;
// Add something so we can see it - will be replaced if there's children
_this . left = jQuery ( "<div>Top / Left</div>" ) . appendTo ( _this . div ) ;
_this . right = jQuery ( "<div>Bottom / Right</div>" ) . appendTo ( _this . div ) ;
// Deferred object so we can wait for children
_this . loading = jQuery . Deferred ( ) ;
// Flag to temporarily ignore resizing
_this . stop _resize = false ;
return _this ;
}
et2 _split . prototype . destroy = function ( ) {
// Stop listening
this . left . next ( ) . off ( "mouseup" ) ;
// Destroy splitter, restore children
this . div . trigger ( "destroy" ) ;
// Destroy dynamic full-height
2020-03-16 21:15:38 +01:00
this . dynheight . destroy ( ) ;
2020-03-30 18:28:48 +02:00
_super . prototype . destroy . call ( this ) ;
2020-02-06 16:26:37 +01:00
// Remove placeholder children
if ( this . _children . length == 0 ) {
this . div . empty ( ) ;
}
this . div . remove ( ) ;
} ;
/ * *
* Tap in here to check if we have real children , because all children should be created
* by this point . If there are , replace the placeholders .
* /
et2 _split . prototype . loadFromXML = function ( _node ) {
_super . prototype . loadFromXML . call ( this , _node ) ;
if ( this . _children . length > 0 ) {
if ( this . _children [ 0 ] ) {
this . left . detach ( ) ;
this . left = jQuery ( this . _children [ 0 ] . getDOMNode ( this . _children [ 0 ] ) )
. appendTo ( this . div ) ;
}
if ( this . _children [ 1 ] ) {
this . right . detach ( ) ;
this . right = jQuery ( this . _children [ 1 ] . getDOMNode ( this . _children [ 1 ] ) )
. appendTo ( this . div ) ;
}
}
// Nextmatches (and possibly other "full height" widgets) need to be adjusted
// Trigger the dynamic height thing to re-initialize
for ( var i = 0 ; i < this . _children . length ; i ++ ) {
if ( this . _children [ i ] . dynheight ) {
this . _children [ i ] . dynheight . outerNode = ( i == 0 ? this . left : this . right ) ;
this . _children [ i ] . dynheight . initialized = false ;
}
}
} ;
et2 _split . prototype . doLoadingFinished = function ( ) {
_super . prototype . doLoadingFinished . call ( this ) ;
// Use a timeout to give the children a chance to finish
var self = this ;
window . setTimeout ( function ( ) {
self . _init _splitter ( ) ;
} , 1 ) ;
// Not done yet, but widget will let you know
return this . loading . promise ( ) ;
} ;
/ * *
* Initialize the splitter UI
* Internal .
* /
et2 _split . prototype . _init _splitter = function ( ) {
if ( ! this . isAttached ( ) )
return ;
// Avoid trying to do anything while hidden - it ruins all the calculations
// Try again in resize()
if ( ! this . div . is ( ':visible' ) )
return ;
var options = {
type : this . orientation ,
dock : this . dock _side ,
splitterClass : "et2_split" ,
outline : true ,
eventNamespace : '.et2_split.' + this . id ,
// Default sizes, in case the preference doesn't work
// Splitter would normally just go to 50%, but our deferred loading
// ruins sizing for it
sizeTop : this . dynheight . outerNode . height ( ) / 2 ,
sizeLeft : this . dynheight . outerNode . width ( ) / 2
} ;
var widget = this ;
//Convert pixel size to percent
var pix2per = function ( _size ) {
var per ;
if ( widget . orientation == "v" ) {
per = _size * 100 / widget . dynheight . outerNode . width ( ) ;
}
else {
per = _size * 100 / widget . dynheight . outerNode . height ( ) ;
}
return per . toFixed ( 2 ) + "%" ;
} ;
// Check for position preference, load it in
if ( this . id ) {
var pref = this . egw ( ) . preference ( 'splitter-size-' + this . id , this . egw ( ) . getAppName ( ) ) ;
if ( pref ) {
if ( this . orientation == "v" && pref [ 'sizeLeft' ] < this . dynheight . outerNode . width ( ) ||
this . orientation == "h" && pref [ 'sizeTop' ] < this . dynheight . outerNode . height ( ) ) {
options = jQuery . extend ( options , pref ) ;
this . prefSize = pref [ this . orientation == "v" ? 'sizeLeft' : 'sizeTop' ] ;
}
}
// If there is no preference yet, set it to half size
// Otherwise the right pane gets the fullsize
if ( typeof this . prefSize == 'undefined' || ! this . prefSize ) {
this . prefSize = this . orientation == "v" ? options . sizeLeft : options . sizeTop ;
}
}
// Avoid double init
if ( this . div . hasClass ( options . splitterClass ) ) {
this . div . trigger ( "destroy" ) ;
}
// Initialize splitter
this . div . splitter ( options ) ;
this . div . trigger ( 'resize' , [ options . sizeTop || options . sizeLeft || 0 ] ) ;
// Start docked?
if ( options . dock ) {
if ( options . dock == "bottomDock" && Math . abs ( options . sizeTop - this . div . height ( ) ) < et2 _split . DOCK _TOLERANCE ||
options . dock == "topDock" && options . sizeTop == 0 ||
! this . div . is ( ':visible' ) // Starting docked if hidden simplifies life when resizing
) {
this . dock ( ) ;
}
}
// Add icon to splitter bar
var icon = "ui-icon-grip-"
+ ( this . dock _side ? "solid" : "dotted" ) + "-"
+ ( this . orientation == "h" ? "horizontal" : "vertical" ) ;
jQuery ( document . createElement ( "div" ) )
. addClass ( "ui-icon" )
. addClass ( icon )
. appendTo ( this . left . next ( ) ) ;
// Save preference when size changed
if ( this . id && this . egw ( ) . getAppName ( ) ) {
var self _1 = this ;
this . left . on ( "resize" + options . eventNamespace , function ( e ) {
// Force immediate layout, so proper layout & sizes are available
// for resize(). Chrome defers layout, so current DOM node sizes
// are not available to widgets if they ask.
var display = this . style . display ;
this . style . display = 'none' ;
this . offsetHeight ;
this . style . display = display ;
if ( e . namespace == options . eventNamespace . substr ( 1 ) && ! self _1 . isDocked ( ) ) {
// Store current position in preferences
var size = self _1 . orientation == "v" ? { sizeLeft : self _1 . left . width ( ) } : { sizeTop : self _1 . left . height ( ) } ;
self _1 . prefSize = size [ self _1 . orientation == "v" ? 'sizeLeft' : 'sizeTop' ] ;
var prefInPercent = self _1 . orientation == "v" ? { sizeLeft : pix2per ( size . sizeLeft ) } : { sizeTop : pix2per ( size . sizeTop ) } ;
if ( parseInt ( self _1 . orientation == 'v' ? prefInPercent . sizeLeft : prefInPercent . sizeTop ) < 100 ) {
self _1 . egw ( ) . set _preference ( self _1 . egw ( ) . getAppName ( ) , 'splitter-size-' + self _1 . id , prefInPercent ) ;
}
}
// Ok, update children
self _1 . iterateOver ( function ( widget ) {
// Extra resize would cause stalling chrome
// as resize might confilict with bottom download bar
// in chrome which does a window resize, so better to not
// trigger second resize and leave that to an application
// if it is neccessary.
// Above forcing is not enough for Firefox, defer
window . setTimeout ( jQuery . proxy ( function ( ) { this . resize ( ) ; } , widget ) , 200 ) ;
} , self _1 , et2 _IResizeable ) ;
} ) ;
}
this . loading . resolve ( ) ;
} ;
/ * *
* Implement the et2 _IResizable interface to resize
* /
et2 _split . prototype . resize = function ( ) {
// Avoid doing anything while hidden - check here, and init if needed
if ( this . div . children ( ) . length <= 2 ) {
this . _init _splitter ( ) ;
}
if ( this . dynheight && ! this . stop _resize ) {
var old _1 = { w : this . div . width ( ) , h : this . div . height ( ) } ;
this . dynheight . update ( function ( w , h ) {
if ( this . orientation == "v" ) {
this . left . height ( h ) ;
this . right . height ( h ) ;
if ( this . left . width ( ) + this . right . width ( ) + this . div . find ( '.splitter-bar' ) . outerWidth ( ) < this . div . width ( ) ||
this . left . width ( ) + this . right . width ( ) - this . div . find ( '.splitter-bar' ) . outerWidth ( ) > this . div . width ( ) )
this . div . trigger ( 'resize.et2_split.' + this . id , this . prefSize ) ;
}
if ( this . orientation == "h" ) {
this . left . width ( w ) ;
this . right . width ( w ) ;
if ( this . isDocked ( ) ) {
if ( this . dock _side == "topDock" ) {
this . right . height ( h ) ;
this . left . height ( 0 ) ;
}
else {
this . left . height ( h ) ;
this . right . height ( 0 ) ;
}
}
}
if ( w != old _1 . w || h != old _1 . h ) {
this . div . trigger ( 'resize.et2_split.' + this . id , this . prefSize ) ;
}
} , this ) ;
}
} ;
et2 _split . prototype . getDOMNode = function ( ) {
return this . div [ 0 ] ;
} ;
/ * *
* Set splitter orientation
*
* @ param orient String "v" or "h"
* /
et2 _split . prototype . set _orientation = function ( orient ) {
this . orientation = orient ;
this . _init _splitter ( ) ;
} ;
/ * *
* Set the side for docking
*
* @ param dock String One of leftDock , rightDock , topDock , bottomDock
* /
et2 _split . prototype . set _dock _side = function ( dock ) {
this . dock _side = dock ;
this . _init _splitter ( ) ;
} ;
/ * *
* Turn on or off resizing while dragging
*
* @ param outline boolean
* /
et2 _split . prototype . set _outline = function ( outline ) {
this . outline = outline ;
this . _init _splitter ( ) ;
} ;
/ * *
* If the splitter has a dock direction set , dock it .
* Docking requires the dock attribute to be set .
* /
et2 _split . prototype . dock = function ( ) {
if ( this . isDocked ( ) )
return ;
this . div . trigger ( "dock" ) ;
} ;
/ * *
* If the splitter is docked , restore it to previous size
* Docking requires the dock attribute to be set .
* /
et2 _split . prototype . undock = function ( ) {
this . div . trigger ( "undock" ) ;
} ;
/ * *
* Determine if the splitter is docked
* @ return boolean
* /
et2 _split . prototype . isDocked = function ( ) {
var bar = jQuery ( '.splitter-bar' , this . div ) ;
return bar . hasClass ( 'splitter-bar-horizontal-docked' ) || bar . hasClass ( 'splitter-bar-vertical-docked' ) ;
} ;
/ * *
* Toggle the splitter ' s docked state .
* Docking requires the dock attribute to be set .
* /
et2 _split . prototype . toggleDock = function ( ) {
this . div . trigger ( "toggleDock" ) ;
} ;
// Printing
/ * *
* Prepare for printing by stopping all the fuss
*
* /
et2 _split . prototype . beforePrint = function ( ) {
// Resizing causes preference changes & relayouts. Don't do it.
this . stop _resize = true ;
// Add the class, if needed
this . div . addClass ( 'print' ) ;
// Don't return anything, just work normally
} ;
et2 _split . prototype . afterPrint = function ( ) {
this . div . removeClass ( 'print' ) ;
this . stop _resize = false ;
} ;
et2 _split . _attributes = {
"orientation" : {
"name" : "Orientation" ,
"description" : "Horizontal or vertical (v or h)" ,
"default" : "v" ,
"type" : "string"
} ,
"outline" : {
"name" : "Outline" ,
"description" : "Use a 'ghosted' copy of the splitbar and does not resize the panes until the mouse button is released. Reduces flickering or unwanted re-layout during resize" ,
"default" : false ,
"type" : "boolean"
} ,
"dock_side" : {
"name" : "Dock" ,
"description" : "Allow the user to 'Dock' the splitbar to one side of the splitter, essentially hiding one pane and using the entire splitter area for the other pane. One of leftDock, rightDock, topDock, bottomDock." ,
"default" : et2 _no _init ,
"type" : "string"
} ,
"width" : {
"default" : "100%"
} ,
// Not needed
"overflow" : { "ignore" : true } ,
"no_lang" : { "ignore" : true } ,
"rows" : { "ignore" : true } ,
"cols" : { "ignore" : true }
} ;
et2 _split . DOCK _TOLERANCE = 15 ;
return et2 _split ;
} ( et2 _core _DOMWidget _1 . et2 _DOMWidget ) ) ;
et2 _core _widget _1 . et2 _register _widget ( et2 _split , [ "split" ] ) ;
//# sourceMappingURL=et2_widget_split.js.map