diff --git a/t3lib/core_autoload.php b/t3lib/core_autoload.php
index 4b4488744681175b7a3a264365418ad11f2aba7b..883285c15a2fac7cd1d5c1bb700c6b46c6cd8c7c 100644
--- a/t3lib/core_autoload.php
+++ b/t3lib/core_autoload.php
@@ -196,16 +196,15 @@ $t3libClasses = array(
 	't3lib_tree_abstractdataprovider' => PATH_t3lib . 'tree/class.t3lib_tree_abstractdataprovider.php',
 	't3lib_tree_abstractstateprovider' => PATH_t3lib . 'tree/class.t3lib_tree_abstractstateprovider.php',
 	't3lib_tree_node' => PATH_t3lib . 'tree/class.t3lib_tree_node.php',
-	't3lib_tree_representationnode' => PATH_t3lib . 'tree/class.t3lib_tree_representationnode.php',
 	't3lib_tree_nodecollection' => PATH_t3lib . 'tree/class.t3lib_tree_nodecollection.php',
 	't3lib_tree_sortednodecollection' => PATH_t3lib . 'tree/class.t3lib_tree_sortednodecollection.php',
-	't3lib_tree_extdirect_abstractextjstree' => PATH_t3lib . 'tree/extdirect/class.t3lib_tree_extdirect_abstractextjstree.php',
-	't3lib_tree_comparablenode' => PATH_t3lib . 'interfaces/tree/interface.t3lib_tree_comparablenode.php',
-	't3lib_tree_draggableanddropable' => PATH_t3lib . 'interfaces/tree/interface.t3lib_tree_draggableanddropable.php',
-	't3lib_tree_labeleditable' => PATH_t3lib . 'interfaces/tree/interface.t3lib_tree_labeleditable.php',
-	't3lib_tree_renderer_abstract' => PATH_t3lib . 'tree/renderer/class.t3lib_tree_renderer_abstract.php',
-	't3lib_tree_renderer_unorderedlist' => PATH_t3lib . 'tree/renderer/class.t3lib_tree_renderer_unorderedlist.php',
-	't3lib_tree_renderer_extjsjson' => PATH_t3lib . 'tree/renderer/class.t3lib_tree_renderer_extjsjson.php',
+	't3lib_tree_extjs_abstractextjstree' => PATH_t3lib . 'tree/extjs/class.t3lib_tree_extjs_abstractextjstree.php',
+	't3lib_tree_extjs_abstractstatefulextjstree' => PATH_t3lib . 'tree/extjs/class.t3lib_tree_extjs_abstractstatefulextjstree.php',
+	't3lib_tree_comparablenode' => PATH_t3lib . 'tree/interfaces/interface.t3lib_tree_comparablenode.php',
+	't3lib_tree_renderablenode' => PATH_t3lib . 'tree/interfaces/interface.t3lib_tree_renderablenode.php',
+	't3lib_tree_draggableanddropable' => PATH_t3lib . 'tree/interfaces/interface.t3lib_tree_draggableanddropable.php',
+	't3lib_tree_labeleditable' => PATH_t3lib . 'tree/interfaces/interface.t3lib_tree_labeleditable.php',
+	't3lib_tree_recordbasednode' => PATH_t3lib . 'tree/interfaces/interface.t3lib_tree_recordbasednode.php',
 	't3lib_contextmenu_abstractcontextmenu' => PATH_t3lib . 'contextmenu/class.t3lib_contextmenu_abstractcontextmenu.php',
 	't3lib_contextmenu_abstractdataprovider' => PATH_t3lib . 'contextmenu/class.t3lib_contextmenu_abstractdataprovider.php',
 	't3lib_contextmenu_action' => PATH_t3lib . 'contextmenu/class.t3lib_contextmenu_action.php',
@@ -225,7 +224,8 @@ $t3libClasses = array(
 	't3lib_contextmenu_pagetree_extdirect_contextmenu' => PATH_t3lib . 'contextmenu/pagetree/extdirect/class.t3lib_contextmenu_pagetree_extdirect_contextmenu.php',
 	't3lib_tree_pagetree_dataprovider' => PATH_t3lib . 'tree/pagetree/class.t3lib_tree_pagetree_dataprovider.php',
 	't3lib_tree_pagetree_node' => PATH_t3lib . 'tree/pagetree/class.t3lib_tree_pagetree_node.php',
-	't3lib_tree_extdirect_node' => PATH_t3lib . 'tree/extdirect/class.t3lib_tree_extdirect_node.php',
+	't3lib_tree_extjs_node' => PATH_t3lib . 'tree/extjs/class.t3lib_tree_extjs_node.php',
+	't3lib_tree_navigationtree_node' => PATH_t3lib . 'tree/navigationtree/class.t3lib_tree_navigationtree_node.php',
 	't3lib_tree_pagetree_nodecollection' => PATH_t3lib . 'tree/pagetree/class.t3lib_tree_pagetree_nodecollection.php',
 	't3lib_tree_pagetree_commands' => PATH_t3lib . 'tree/pagetree/class.t3lib_tree_pagetree_commands.php',
 	't3lib_contextmenu_pagetree_dataprovider' => PATH_t3lib . 'contextmenu/pagetree/class.t3lib_contextmenu_pagetree_dataprovider.php',
@@ -238,4 +238,4 @@ $t3libClasses = array(
 $tslibClasses = require(PATH_typo3 . 'sysext/cms/ext_autoload.php');
 
 return array_merge($t3libClasses, $tslibClasses);
-?>
\ No newline at end of file
+?>
diff --git a/t3lib/js/extjs/components/pagetree/javascript/Ext.ux.state.TreePanel.js b/t3lib/js/extjs/components/pagetree/javascript/Ext.ux.state.TreePanel.js
index 348d94cb900ae39cf8189e959c2ffba88c761b13..a781831f2d4b3b2ba5b26e5e98e7d9a70a0eb4c9 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/Ext.ux.state.TreePanel.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/Ext.ux.state.TreePanel.js
@@ -23,126 +23,117 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.ns('Ext.ux.state');
-
-// dummy constructor
-Ext.ux.state.TreePanel = function() {};
-
 /**
  * State Provider for a tree panel
  */
-Ext.override(Ext.ux.state.TreePanel, {
+Ext.define('Ext.ux.state.TreePanel', {
+	extend: 'Ext.state.Stateful',
+
 	/**
-	 * Initializes the plugin
-	 * @param {Ext.tree.TreePanel} tree
-	 * @private
+	 * Mixin constructor
+	 * @param {Object} config
+	 *
 	 */
-	init:function(tree) {
-		tree.lastSelectedNode = null;
-		tree.isRestoringState = false;
-		tree.stateHash = {};
-
-		// install event handlers on TreePanel
-		tree.on({
-			// add path of expanded node to stateHash
-			 beforeexpandnode:function(n) {
-				if (this.isRestoringState) {
-					return;
-				}
-				var saveID = n.id;
-				this.stateHash[saveID.substr(1)] = 1;
-			},
-
-			// delete path and all subpaths of collapsed node from stateHash
-			beforecollapsenode:function(n) {
-				if (this.isRestoringState) {
-					return;
-				}
-
-				var deleteID = n.id;
-				delete this.stateHash[deleteID.substr(1)];
-			},
-
-			beforeclick: function(node) {
-				if (this.isRestoringState) {
-					return;
-				}
-				this.stateHash['lastSelectedNode'] = node.id;
-			}
-		});
-
-			// update state on node expand or collapse
-		tree.stateEvents = tree.stateEvents || [];
-		tree.stateEvents.push('expandnode', 'collapsenode', 'click');
-
-		// add state related props to the tree
-		Ext.apply(tree, {
-			// keeps expanded nodes paths keyed by node.ids
-			stateHash:{},
-
-			restoreState: function() {
-				this.isRestoringState = true;
-					// get last selected node
-				for (var pageID in this.stateHash) {
-					var pageNode = this.getNodeById('p' + pageID);
-					if (pageNode) {
-						pageNode.on({
-							expand: {
-								single:true,
-								scope:this,
-								fn: this.restoreState
+	constructor: function (config) {
+		if (config.stateful) {
+				// Install event handlers on TreePanel
+			this.on({
+					// Add path of expanded node to stateHash
+				beforeitemexpand: function (node) {
+						if (this.isRestoringState) {
+							return;
+						}
+						this.stateHash[String(node.getNodeData('id'))] = node.getId();
+					},
+					// Delete collapsed node from stateHash
+				beforeitemcollapse: function (node) {
+						if (this.isRestoringState) {
+							return;
+						}
+						delete this.stateHash[String(node.getNodeData('id'))];
+					},
+					// Update last selected node in stateHash
+				selectionchange: function (view, node) {
+					this.onStateChange();
+						if (this.isRestoringState) {
+							return;
+						}
+						if (this.getSelectionModel().getLastSelected()) {
+							this.stateHash['lastSelectedNode'] = this.getSelectionModel().getLastSelected().getId();
+						}
+					},
+				scope: this
+			});
+				// Add/override state properties
+			Ext.apply(this, {
+					// Update state on node expand, collapse or selection change
+				stateEvents: ['itemexpand', 'itemcollapse', 'selectionchange'],
+					// Avoid updating state while restoring
+				isRestoringState: false,
+					// State object
+				stateHash: {},
+				/**
+				 * Restore tree state into the saved state
+				 *
+				 */
+				restoreState: function() {
+					if (this.stateful) {
+						this.isRestoringState = true;
+							// Expand paths according to stateHash
+						for (var pageID in this.stateHash) {
+							var pageNode = this.getStore().getNodeById(this.stateHash[pageID]);
+							if (pageNode && !pageNode.isLeaf() && !pageNode.isExpanded()) {
+								pageNode.on({
+									expand: {
+										single: true,
+										scope: this,
+										fn: this.restoreState
+									}
+								});
+								pageNode.set('expanded', true);
+								pageNode.commit();
+								this.refreshNode(pageNode);
+								pageNode.fireEvent('expand', pageNode);
 							}
-						});
-						if (pageNode.expanded === false && pageNode.rendered == true) {
-							pageNode.expand();
 						}
-					}
-				}
-				if (this.stateHash['lastSelectedNode']) {
-					var node = this.getNodeById(this.stateHash['lastSelectedNode']);
-					if (node) {
-						this.selectPath(node.getPath());
-
-						var contentId = TYPO3.Backend.ContentContainer.getIdFromUrl() ||
-							String(fsMod.recentIds['web']) || '-1';
-
-						var isCurrentSelectedNode = (
-							String(node.attributes.nodeData.id) === contentId ||
-							contentId.indexOf('pages' + String(node.attributes.nodeData.id)) !== -1
-						);
-
-						if (contentId !== '-1' && !isCurrentSelectedNode && this.app.isVisible() &&
-							this.commandProvider && this.commandProvider.singleClick
-						) {
-							this.commandProvider.singleClick(node, this);
+							// Get last selected node
+						if (this.stateHash['lastSelectedNode']) {
+							var node = this.getStore().getNodeById(this.stateHash['lastSelectedNode']);
+							if (node) {
+								this.selectPath(node.getPath());
+				
+								var contentId = TYPO3.Backend.ContentContainer.getIdFromUrl() ||
+									String(fsMod.recentIds['web']) || '-1';
+				
+								var isCurrentSelectedNode = (
+									String(node.getNodeData('id')) === contentId ||
+									contentId.indexOf('pages' + node.getNodeData('id')) !== -1
+								);
+				
+								if (contentId !== '-1' 
+									&& !isCurrentSelectedNode
+									&& this == this.app.getTree()
+									&& this.commandProvider
+									&& this.commandProvider.singleClick
+								) {
+									this.commandProvider.singleClick(node, this);
+								}
+							}
 						}
+						this.isRestoringState = false;
 					}
+				},
+				/**
+				 * Return stateHash for save by state manager
+				 *
+				 */
+				getState: function() {
+					return {
+						stateHash: this.stateHash
+					};
 				}
-
-				this.isRestoringState = false;
-			},
-
-			// apply state on tree initialization
-			applyState:function(state) {
-				if(state) {
-					Ext.apply(this, state);
-
-					// it is too early to expand paths here
-					// so do it once on root load
-					this.root.on({
-						load: {
-							single:true,
-							scope:this,
-							fn: this.restoreState
-						}
-					});
-				}
-			},
-
-			// returns stateHash for save by state manager
-			getState:function() {
-				return {stateHash:this.stateHash};
-			}
-		});
+			});
+			this.callParent(arguments);
+		}
 	}
 });
diff --git a/t3lib/js/extjs/components/pagetree/javascript/actions.js b/t3lib/js/extjs/components/pagetree/javascript/actions.js
index 1dbb5b03ea961297f5a5429a5781da9b40952087..3b4947450931cc03383d79b235882c1ed85e7069 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/actions.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/actions.js
@@ -61,8 +61,8 @@ TYPO3.Components.PageTree.Actions = {
 		tree.t3ContextInfo.inCopyMode = false;
 
 		if (tree.t3ContextNode) {
-			tree.t3ContextNode.attributes.nodeData.t3InCutMode = false;
-			tree.t3ContextNode.attributes.nodeData.t3InCopyMode = false;
+			tree.t3ContextNode.setNodeData('t3InCutMode', false);
+			tree.t3ContextNode.setNodeData('t3InCopyMode', false);
 			tree.t3ContextNode = null;
 		}
 	},
@@ -71,51 +71,55 @@ TYPO3.Components.PageTree.Actions = {
 	 * Updates an existing node with the given alternative. The new tree node
 	 * is returned afterwards.
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {Boolean} isExpanded
 	 * @param {Object} updatedNode
 	 * @param {Function} callback
-	 * @return {Ext.tree.TreeNode}
+	 * @return {Ext.data.NodeInterface}
 	 */
-	updateNode: function(node, isExpanded, updatedNode, callback) {
+	updateNode: function(tree, node, isExpanded, updatedNode, callback) {
 		if (!updatedNode) {
 			return null;
 		}
 
-		updatedNode.uiProvider = node.ownerTree.uiProvider;
-		var newTreeNode = new Ext.tree.TreeNode(updatedNode);
+		var newTreeNode = Ext.create('TYPO3.Components.PageTree.Model', updatedNode);
 
 		var refreshCallback = this.restoreNodeStateAfterRefresh;
 		if (callback) {
-			refreshCallback = refreshCallback.createSequence(callback);
+			refreshCallback = Ext.Function.createSequence(refreshCallback, callback);
 		}
 
 		node.parentNode.replaceChild(newTreeNode, node);
-		newTreeNode.ownerTree.refreshNode(newTreeNode, refreshCallback);
-
+		newTreeNode.set('expanded', isExpanded);
+		tree.refreshNode(newTreeNode, refreshCallback, tree);
 		return newTreeNode;
 	},
 
 	/**
 	 * Restores the node state
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {Boolean} isExpanded
 	 * @return {void}
 	 */
-	restoreNodeStateAfterRefresh: function(node, isExpanded) {
-		node.parentNode.expand(false, false);
-		if (isExpanded) {
-			node.expand(false, false);
-		} else {
-			node.collapse(false, false);
+	restoreNodeStateAfterRefresh: function (records, operation, successful) {
+		if (successful && operation.node) {
+			var node = operation.node;
+			node.parentNode.expand(false);
+			if (node.isExpanded()) {
+				node.expand(false);
+			} else {
+				node.collapse(false);
+			}
 		}
 	},
 
 	/**
 	 * Shows deletion confirmation window
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @param {Function} callback
 	 * @param {Boolean} recursiveDelete
@@ -130,10 +134,10 @@ TYPO3.Components.PageTree.Actions = {
 			message = TYPO3.Components.PageTree.LLL.recursiveDeleteDialogMessage;
 		}
 
-		Ext.Msg.show({
+		Ext.MessageBox.show({
 			title: title,
 			msg: message,
-			buttons: Ext.Msg.YESNO,
+			buttons: Ext.MessageBox.YESNO,
 			fn: function (answer) {
 				if (answer === 'yes') {
 					TYPO3.Components.PageTree.Actions.deleteNode(node, tree, callback);
@@ -141,21 +145,21 @@ TYPO3.Components.PageTree.Actions = {
 				}
 				return false;
 			},
-			animEl: 'elId'
+			animateTarget: tree.getView().getNode(node).id
 		});
 	},
 
 	/**
 	 * Deletes a node directly
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @param {Function} callback
 	 * @return {void}
 	 */
 	deleteNode: function(node, tree, callback) {
 		TYPO3.Components.PageTree.Commands.deleteNode(
-			node.attributes.nodeData,
+			node.get('nodeData'),
 			function(response) {
 				var succeeded = this.evaluateResponse(response);
 				if (Ext.isFunction(callback)) {
@@ -165,7 +169,7 @@ TYPO3.Components.PageTree.Actions = {
 				if (succeeded) {
 						// the node may not be removed in workspace mode
 					if (top.TYPO3.configuration.inWorkspace && response.id) {
-						this.updateNode(node, node.isExpanded(), response);
+						this.updateNode(tree, node, node.isExpanded(), response);
 					} else {
 						node.remove();
 					}
@@ -178,13 +182,13 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Removes a node either directly or first shows deletion popup
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	removeNode: function(node, tree) {
 		if (TYPO3.Components.PageTree.Configuration.displayDeleteConfirmation) {
-			this.confirmDelete(node);
+			this.confirmDelete(node, tree);
 		} else {
 			this.deleteNode(node, tree);
 		}
@@ -194,19 +198,19 @@ TYPO3.Components.PageTree.Actions = {
 	 * Restores a given node and moves it to the given destination inside the tree. Use this
 	 * method if you want to add it as the first child of the destination.
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {Ext.data.NodeInterface} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
-	 * @param {Ext.tree.TreeNode} destination
+	 * @param {Ext.data.NodeInterface} destination
 	 * @return {void}
 	 */
 	restoreNodeToFirstChildOfDestination: function(node, tree, destination) {
 		TYPO3.Components.PageTree.Commands.restoreNode(
-			node.attributes.nodeData,
-			destination.attributes.nodeData.id,
-			function(updatedNode) {
+			node.get('nodeData'),
+			destination.get('nodeData').id,
+			function (updatedNode) {
 				if (this.evaluateResponse(updatedNode)) {
-					var newTreeNode = new Ext.tree.TreeNode(
-						Ext.apply(node.attributes, updatedNode)
+					var newTreeNode = Ext.create('TYPO3.Components.PageTree.Model',
+						Ext.apply(node.fields, updatedNode)
 					);
 					destination.insertBefore(newTreeNode, destination.firstChild);
 				}
@@ -219,19 +223,19 @@ TYPO3.Components.PageTree.Actions = {
 	 * Restores a given node and moves it to the given destination inside the tree. Use this
 	 * method if you want to add the node after the destination node.
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
-	 * @param {Ext.tree.TreeNode} destination
+	 * @param {TYPO3.Components.PageTree.Model} destination
 	 * @return {void}
 	 */
 	restoreNodeAfterDestination: function(node, tree, destination) {
 		TYPO3.Components.PageTree.Commands.restoreNode(
-			node.attributes.nodeData,
-			-destination.attributes.nodeData.id,
-			function(updatedNode) {
+			node.get('nodeData'),
+			-destination.get('nodeData').id,
+			function (updatedNode) {
 				if (this.evaluateResponse(updatedNode)) {
-					var newTreeNode = new Ext.tree.TreeNode(
-						Ext.apply(node.attributes, updatedNode)
+					var newTreeNode = Ext.create('TYPO3.Components.PageTree.Model',
+						Ext.apply(node.fields, updatedNode)
 					);
 					destination.parentNode.insertBefore(newTreeNode, destination.nextSibling);
 				}
@@ -243,7 +247,7 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Collapses a whole tree branch
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @return {void}
 	 */
 	collapseBranch: function(node) {
@@ -253,7 +257,7 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Expands a whole tree branch
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @return {void}
 	 */
 	expandBranch: function(node) {
@@ -263,13 +267,13 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Opens a popup windows for previewing the given node/page
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @return {void}
 	 */
 	viewPage: function(node) {
 		var frontendWindow = window.open('', 'newTYPO3frontendWindow');
 		TYPO3.Components.PageTree.Commands.getViewLink(
-			node.attributes.nodeData,
+			node.get('nodeData'),
 			function(result) {
 				frontendWindow.location = result;
 				frontendWindow.focus();
@@ -280,13 +284,13 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Creates a temporary tree mount point
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	mountAsTreeRoot: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.setTemporaryMountPoint(
-			node.attributes.nodeData,
+			node.get('nodeData'),
 			function(response) {
 				if (TYPO3.Components.PageTree.Configuration.temporaryMountPoint) {
 					TYPO3.Backend.NavigationContainer.PageTree.removeIndicator(
@@ -297,19 +301,18 @@ TYPO3.Components.PageTree.Actions = {
 				TYPO3.Components.PageTree.Configuration.temporaryMountPoint = response;
 				TYPO3.Backend.NavigationContainer.PageTree.addTemporaryMountPointIndicator();
 
-				var selectedNode = TYPO3.Backend.NavigationContainer.PageTree.getSelected();
+				var selectedNode = tree.getSelectionModel().getLastSelected();
 				tree.stateId = 'Pagetree' + TYPO3.Components.PageTree.Configuration.temporaryMountPoint;
-				tree.refreshTree(function() {
-					var nodeIsSelected = false;
+					// Refresh tree and restore last selection if under the temporary mount
+				tree.refreshTree(function () {
+					var selectionModel = tree.getSelectionModel();
 					if (selectedNode) {
-						nodeIsSelected = TYPO3.Backend.NavigationContainer.PageTree.select(
-							selectedNode.attributes.nodeData.id
-						);
+						selectionModel.deselect(selectedNode);
+						selectionModel.select(selectedNode);
 					}
-
-					var node = (nodeIsSelected ? TYPO3.Backend.NavigationContainer.PageTree.getSelected() : null);
-					if (node) {
-						this.singleClick(node, tree);
+					selectedNode = selectionModel.getLastSelected();
+					if (selectedNode) {
+						this.singleClick(selectedNode, tree);
 					} else {
 						this.singleClick(tree.getRootNode().firstChild, tree);
 					}
@@ -322,63 +325,67 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Opens the edit page properties dialog
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	editPageProperties: function(node) {
-		node.select();
+	editPageProperties: function(node, tree) {
+		tree.getSelectionModel().select(node);
 		TYPO3.Backend.ContentContainer.setUrl(
-			'alt_doc.php?edit[pages][' + node.attributes.nodeData.id + ']=edit'
+			'alt_doc.php?edit[pages][' + node.getNodeData('id') + ']=edit'
 		);
 	},
 
 	/**
 	 * Opens the new page wizard
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	newPageWizard: function(node) {
-		node.select();
+	newPageWizard: function(node, tree) {
+		tree.getSelectionModel().select(node);
 		TYPO3.Backend.ContentContainer.setUrl(
-			'db_new.php?id=' + node.attributes.nodeData.id + '&pagesOnly=1'
+			'db_new.php?id=' + node.getNodeData('id') + '&pagesOnly=1'
 		);
 	},
 
 	/**
 	 * Opens the info popup
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @return {void}
 	 */
 	openInfoPopUp: function(node) {
-		launchView('pages', node.attributes.nodeData.id);
+		launchView('pages', node.getNodeData('id'));
 	},
 
 	/**
 	 * Opens the history popup
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree	 
 	 * @return {void}
 	 */
-	openHistoryPopUp: function(node) {
-		node.select();
+	openHistoryPopUp: function(node, tree) {
+		tree.getSelectionModel().select(node);
 		TYPO3.Backend.ContentContainer.setUrl(
-			'show_rechis.php?element=pages:' + node.attributes.nodeData.id
+			'show_rechis.php?element=pages:' + node.getNodeData('id')
 		);
 	},
 
 	/**
 	 * Opens the export .t3d file dialog
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	exportT3d: function(node) {
-		node.select();
+	exportT3d: function(node, tree) {
+		tree.getSelectionModel().select(node);
 		TYPO3.Backend.ContentContainer.setUrl(
 			'sysext/impexp/app/index.php?tx_impexp[action]=export&' +
-				'id=0&tx_impexp[pagetree][id]=' + node.attributes.nodeData.id +
+				'id=0&tx_impexp[pagetree][id]=' + node.getNodeData('id') +
 				'&tx_impexp[pagetree][levels]=0' +
 				'&tx_impexp[pagetree][tables][]=_ALL'
 		);
@@ -387,13 +394,14 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Opens the import .t3d file dialog
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree 
 	 * @return {void}
 	 */
-	importT3d: function(node) {
-		node.select();
+	importT3d: function(node, tree) {
+		tree.getSelectionModel().select(node);
 		TYPO3.Backend.ContentContainer.setUrl(
-			'sysext/impexp/app/index.php?id=' + node.attributes.nodeData.id +
+			'sysext/impexp/app/index.php?id=' + node.getNodeData('id') +
 				'&table=pages&tx_impexp[action]=import'
 		);
 	},
@@ -401,13 +409,13 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Enables the cut mode of a node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	enableCutMode: function(node, tree) {
 		this.disableCopyMode(node, tree);
-		node.attributes.nodeData.t3InCutMode = true;
+		node.setNodeData('t3InCutMode', true);
 		tree.t3ContextInfo.inCutMode = true;
 		tree.t3ContextNode = node;
 	},
@@ -415,25 +423,25 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Disables the cut mode of a node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	disableCutMode: function(node, tree) {
 		this.releaseCutAndCopyModes(tree);
-		node.attributes.nodeData.t3InCutMode = false;
+		node.setNodeData('t3InCutMode', false);
 	},
 
 	/**
 	 * Enables the copy mode of a node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	enableCopyMode: function(node, tree) {
 		this.disableCutMode(node, tree);
-		node.attributes.nodeData.t3InCopyMode = true;
+		node.setNodeData('t3InCopyMode', true);
 		tree.t3ContextInfo.inCopyMode = true;
 		tree.t3ContextNode = node;
 	},
@@ -441,19 +449,19 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Disables the copy mode of a node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	disableCopyMode: function(node, tree) {
 		this.releaseCutAndCopyModes(tree);
-		node.attributes.nodeData.t3InCopyMode = false;
+		node.setNodeData('t3InCopyMode', false);
 	},
 
 	/**
 	 * Pastes the cut/copy context node into the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
@@ -463,10 +471,10 @@ TYPO3.Components.PageTree.Actions = {
 		}
 
 		if (tree.t3ContextInfo.inCopyMode) {
-			var newNode = tree.t3ContextNode = new Ext.tree.TreeNode(tree.t3ContextNode.attributes);
+			var newNode = tree.t3ContextNode = Ext.create('TYPO3.Components.PageTree.Model', tree.t3ContextNode.fields);
 			newNode.id = 'fakeNode';
 			node.insertBefore(newNode, node.childNodes[0]);
-			node.attributes.nodeData.t3InCopyMode = false;
+			node.setNodeData('t3InCopyMode', false);
 			this.copyNodeToFirstChildOfDestination(newNode, tree);
 
 		} else if (tree.t3ContextInfo.inCutMode) {
@@ -475,7 +483,7 @@ TYPO3.Components.PageTree.Actions = {
 			}
 
 			node.appendChild(tree.t3ContextNode);
-			node.attributes.nodeData.t3InCutMode = false;
+			node.setNodeData('t3InCutMode', false);
 			this.moveNodeToFirstChildOfDestination(node, tree);
 		}
 	},
@@ -483,7 +491,7 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Pastes a cut/copy context node after the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
@@ -493,19 +501,18 @@ TYPO3.Components.PageTree.Actions = {
 		}
 
 		if (tree.t3ContextInfo.inCopyMode) {
-			var newNode = tree.t3ContextNode = new Ext.tree.TreeNode(tree.t3ContextNode.attributes);
+			var newNode = tree.t3ContextNode = Ext.create('TYPO3.Components.PageTree.Model', tree.t3ContextNode.fields);
 			newNode.id = 'fakeNode';
 			node.parentNode.insertBefore(newNode, node.nextSibling);
-			node.attributes.nodeData.t3InCopyMode = false;
+			node.setNodeData('t3InCopyMode', false);
 			this.copyNodeAfterDestination(newNode, tree);
 
 		} else if (tree.t3ContextInfo.inCutMode) {
 			if (node.getPath().indexOf(tree.t3ContextNode.id) !== -1) {
 				return;
 			}
-
 			node.parentNode.insertBefore(tree.t3ContextNode, node.nextSibling);
-			node.attributes.nodeData.t3InCutMode = false;
+			node.setNodeData('t3InCutMode', false);
 			this.moveNodeAfterDestination(node, tree);
 		}
 	},
@@ -513,17 +520,17 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Moves the current tree context node after the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	moveNodeAfterDestination: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.moveNodeAfterDestination(
-			tree.t3ContextNode.attributes.nodeData,
-			node.attributes.nodeData.id,
-			function(response) {
+			tree.t3ContextNode.get('nodeData'),
+			node.getNodeData('id'),
+			function (response) {
 				if (this.evaluateResponse(response) && tree.t3ContextNode) {
-					this.updateNode(tree.t3ContextNode, tree.t3ContextNode.isExpanded(), response);
+					this.updateNode(tree, tree.t3ContextNode, tree.t3ContextNode.isExpanded(), response);
 				}
 				this.releaseCutAndCopyModes(tree);
 			},
@@ -534,17 +541,17 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Moves the current tree context node as the first child of the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	moveNodeToFirstChildOfDestination: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.moveNodeToFirstChildOfDestination(
-			tree.t3ContextNode.attributes.nodeData,
-			node.attributes.nodeData.id,
-			function(response) {
+			tree.t3ContextNode.get('nodeData'),
+			node.getNodeData('id'),
+			function (response) {
 				if (this.evaluateResponse(response) && tree.t3ContextNode) {
-					this.updateNode(tree.t3ContextNode, tree.t3ContextNode.isExpanded(), response);
+					this.updateNode(tree, tree.t3ContextNode, tree.t3ContextNode.isExpanded(), response);
 				}
 				this.releaseCutAndCopyModes(tree);
 			},
@@ -555,19 +562,21 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Inserts a new node after the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	insertNodeAfterDestination: function(node, tree) {
+	insertNodeAfterDestination: function (node, tree) {
 		TYPO3.Components.PageTree.Commands.insertNodeAfterDestination(
-			tree.t3ContextNode.attributes.nodeData,
-			node.previousSibling.attributes.nodeData.id,
+			tree.t3ContextNode.get('nodeData'),
+			node.previousSibling.getNodeData('id'),
 			tree.t3ContextInfo.serverNodeType,
-			function(response) {
+			function (response) {
 				if (this.evaluateResponse(response)) {
-					this.updateNode(node, node.isExpanded(), response, function(node) {
-						tree.triggerEdit(node);
+					this.updateNode(tree, node, node.isExpanded(), response, function (records, operation, successful) {
+						if (successful && operation.node) {
+							tree.getPlugin('treeEditor').startEdit(operation.node, tree.getView().getHeaderAtIndex(0));
+						}
 					});
 				}
 				this.releaseCutAndCopyModes(tree);
@@ -579,18 +588,20 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Inserts a new node as the first child of the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	insertNodeToFirstChildOfDestination: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.insertNodeToFirstChildOfDestination(
-			tree.t3ContextNode.attributes.nodeData,
+			tree.t3ContextNode.get('nodeData'),
 			tree.t3ContextInfo.serverNodeType,
-			function(response) {
+			function (response) {
 				if (this.evaluateResponse(response)) {
-					this.updateNode(node, true, response, function(node) {
-						tree.triggerEdit(node);
+					this.updateNode(tree, node, true, response, function (records, operation, successful) {
+						if (successful && operation.node) {
+							tree.getPlugin('treeEditor').startEdit(operation.node, tree.getView().getHeaderAtIndex(0));
+						}
 					});
 				}
 				this.releaseCutAndCopyModes(tree);
@@ -602,18 +613,20 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Copies the current tree context node after the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	copyNodeAfterDestination: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.copyNodeAfterDestination(
-			tree.t3ContextNode.attributes.nodeData,
-			node.previousSibling.attributes.nodeData.id,
-			function(response) {
+			tree.t3ContextNode.get('nodeData'),
+			node.previousSibling.getNodeData('id'),
+			function (response) {
 				if (this.evaluateResponse(response)) {
-					this.updateNode(node, true, response, function(node) {
-						tree.triggerEdit(node);
+					this.updateNode(tree, node, true, response, function (records, operation, successful) {
+						if (successful && operation.node) {
+							tree.getPlugin('treeEditor').startEdit(operation.node, tree.getView().getHeaderAtIndex(0));
+						}
 					});
 				}
 				this.releaseCutAndCopyModes(tree);
@@ -625,18 +638,20 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Copies the current tree context node as the first child of the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
 	copyNodeToFirstChildOfDestination: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.copyNodeToFirstChildOfDestination(
-			tree.t3ContextNode.attributes.nodeData,
-			node.parentNode.attributes.nodeData.id,
-			function(response) {
+			tree.t3ContextNode.get('nodeData'),
+			node.parentNode.getNodeData('id'),
+			function (response) {
 				if (this.evaluateResponse(response)) {
-					this.updateNode(node, true, response, function(node) {
-						tree.triggerEdit(node);
+					this.updateNode(tree, node, true, response, function (records, operation, successful) {
+						if (successful && operation.node) {
+							tree.getPlugin('treeEditor').startEdit(operation.node, tree.getView().getHeaderAtIndex(0));
+						}
 					});
 				}
 				this.releaseCutAndCopyModes(tree);
@@ -648,15 +663,16 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Visibilizes a page
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {Ext.data.NodeInterface} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	enablePage: function(node) {
+	enablePage: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.visiblyNode(
-			node.attributes.nodeData,
+			node.get('nodeData'),
 			function(response) {
 				if (this.evaluateResponse(response)) {
-					this.updateNode(node, node.isExpanded(), response);
+					this.updateNode(tree, node, node.isExpanded(), response);
 				}
 			},
 			this
@@ -666,15 +682,16 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Disables a page
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {Ext.data.NodeInterface} node
+	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	disablePage: function(node) {
+	disablePage: function(node, tree) {
 		TYPO3.Components.PageTree.Commands.disableNode(
-			node.attributes.nodeData,
-			function(response) {
+			node.get('nodeData'),
+			function (response) {
 				if (this.evaluateResponse(response)) {
-					this.updateNode(node, node.isExpanded(), response);
+					this.updateNode(tree, node, node.isExpanded(), response);
 				}
 			},
 			this
@@ -684,11 +701,11 @@ TYPO3.Components.PageTree.Actions = {
 	/**
 	 * Reloads the content frame with the current module and node id
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	singleClick: function(node, tree) {
+	singleClick: function (node, tree) {
 		tree.currentSelectedNode = node;
 
 		var separator = '?';
@@ -696,68 +713,77 @@ TYPO3.Components.PageTree.Actions = {
 			separator = '&';
 		}
 
-		node.select();
-		if (tree.stateHash) {
-			tree.stateHash.lastSelectedNode = node.id;
-		}
+		tree.getSelectionModel().select(node);
 
-		fsMod.recentIds['web'] = node.attributes.nodeData.id;
+		fsMod.recentIds['web'] = node.getNodeData('id');
 
 		TYPO3.Backend.ContentContainer.setUrl(
-			TS.PATH_typo3 + currentSubScript + separator + 'id=' + node.attributes.nodeData.id
+			TS.PATH_typo3 + currentSubScript + separator + 'id=' + node.getNodeData('id')
 		);
 	},
 
 	/**
 	 * Opens a configured url inside the content frame
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @param {Object} contextItem
 	 * @return {void}
 	 */
-	openCustomUrlInContentFrame: function(node, tree, contextItem) {
+	openCustomUrlInContentFrame: function (node, tree, contextItem) {
 		if (!contextItem.customAttributes || !contextItem.customAttributes.contentUrl) {
 			return;
 		}
 
-		node.select();
+		tree.getSelectionModel().select(node);
 		TYPO3.Backend.ContentContainer.setUrl(
-			contextItem.customAttributes.contentUrl.replace('###ID###', node.attributes.nodeData.id)
+			contextItem.customAttributes.contentUrl.replace('###ID###', node.getNodeData('id'))
 		);
 	},
 
 	/**
 	 * Updates the title of a node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {String} newText
 	 * @param {String} oldText
 	 * @param {TYPO3.Components.PageTree.TreeEditor} treeEditor
+	 * @param {TYPO3.Components.PageTree.Tree} tree
+	 * @param {String} dataIndex
 	 * @return {void}
 	 */
-	saveTitle: function(node, newText, oldText, treeEditor) {
-		this.singleClick(node.editNode, node.editNode.ownerTree);
+	saveTitle: function (node, newText, oldText, treeEditor, tree, dataIndex) {
+		this.singleClick(node, tree);
 		if (newText === oldText || newText == '') {
 			treeEditor.updateNodeText(
 				node,
-				node.editNode.attributes.nodeData.editableText,
-				Ext.util.Format.htmlEncode(oldText)
+				node.getNodeData('editableText'),
+				Ext.util.Format.htmlEncode(oldText),
+				dataIndex,
+				tree
 			);
 			return;
 		}
 
 		TYPO3.Components.PageTree.Commands.updateLabel(
-			node.editNode.attributes.nodeData,
+			node.get('nodeData'),
 			newText,
 			function(response) {
 				if (this.evaluateResponse(response)) {
-					treeEditor.updateNodeText(node, response.editableText, response.updatedText);
+					treeEditor.updateNodeText(
+						node,
+						response.editableText,
+						response.updatedText,
+						dataIndex,
+						tree
+					);
 				} else {
 					treeEditor.updateNodeText(
 						node,
-						node.editNode.attributes.nodeData.editableText,
-						Ext.util.Format.htmlEncode(oldText)
+						node.getNodeData('editableText'),
+						Ext.util.Format.htmlEncode(oldText),
+						dataIndex,
+						tree
 					);
 				}
 			},
diff --git a/t3lib/js/extjs/components/pagetree/javascript/app.js b/t3lib/js/extjs/components/pagetree/javascript/app.js
index 6a178f5921a17fdeef4d826158e41a9b98d92f71..2701f93c47a6ed39f31b13bb1ed24f410f2684aa 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/app.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/app.js
@@ -23,18 +23,18 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.App
  *
  * Page tree main application that controls setups the components
  *
  * @namespace TYPO3.Components.PageTree
- * @extends Ext.Panel
+ * @extends Ext.panel.Panel
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
+Ext.define('TYPO3.Components.PageTree.App', {
+	extend: 'Ext.panel.Panel',
+
 	/**
 	 * Panel id
 	 *
@@ -45,7 +45,7 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	/**
 	 * Border
 	 *
-	 * @type {Boolean}
+	 * @type {String}
 	 */
 	border: false,
 
@@ -54,14 +54,38 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 *
 	 * @type {String}
 	 */
-	layout:'fit',
+	layout: 'border',
 
 	/**
-	 * Active tree
+	 * Components defaults
 	 *
-	 * @type {TYPO3.Components.PageTree.Tree}
+	 * @type {Object}
 	 */
-	activeTree: null,
+	 defaults: {
+	 	 border: false,
+	 },
+	
+	/**
+	 * Tree containe
+	 *
+	 * @type {String}
+	 */
+	 items: [{
+			itemId: 'topPanelItems',
+			height: 49,
+			region: 'north',
+			cls: 'typo3-pageTree-topPanelItems'
+	 	},{
+			itemId: 'treeContainer',
+			autoScroll: false,
+			layout: 'card',
+			region: 'center'
+		},{
+			xtype: 'container',
+			itemId: 'deletionDropZoneContainer',
+			cls: 'typo3-pageTree-deletionDropZoneContainer',
+			region: 'south'
+	 }],
 
 	/**
 	 * Initializes the application
@@ -72,112 +96,89 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	initComponent: function() {
-		TYPO3.Components.PageTree.DataProvider.loadResources(function(response) {
-			TYPO3.Components.PageTree.LLL = response['LLL'];
-			TYPO3.Components.PageTree.Configuration = response['Configuration'];
-			TYPO3.Components.PageTree.Sprites = response['Sprites'];
-
-			var tree = this.activeTree = new TYPO3.Components.PageTree.Tree({
-				id: this.id + '-tree',
-				deletionDropZoneId: this.id + '-deletionDropZone',
-				ddGroup: this.id,
-				stateful: true,
-				stateId: 'Pagetree' + TYPO3.Components.PageTree.Configuration.temporaryMountPoint,
-				stateEvents: [],
-				autoScroll: true,
-				autoHeight: false,
-				plugins: new Ext.ux.state.TreePanel(),
-				commandProvider: TYPO3.Components.PageTree.Actions,
-				contextMenuProvider: TYPO3.Components.PageTree.ContextMenuDataProvider,
-				treeDataProvider: TYPO3.Components.PageTree.DataProvider,
-				app: this,
-				listeners: {
-					resize: {
-						fn: function() {
-							var treeContainer = Ext.getCmp(this.id + '-treeContainer');
-							Ext.getCmp(this.id + '-filteringTree').setSize(treeContainer.getSize());
-							treeContainer.doLayout();
-						},
-						scope: this,
-						buffer: 250
-					}
-				}
-			});
-
-			var filteringTree = new TYPO3.Components.PageTree.FilteringTree({
-				id: this.id + '-filteringTree',
-				deletionDropZoneId: this.id + '-deletionDropZone',
-				ddGroup: this.id,
-				autoScroll: true,
-				autoHeight: false,
-				commandProvider: TYPO3.Components.PageTree.Actions,
-				contextMenuProvider: TYPO3.Components.PageTree.ContextMenuDataProvider,
-				treeDataProvider: TYPO3.Components.PageTree.DataProvider,
-				app: this
-			}).hide();
-
-			var topPanel = new TYPO3.Components.PageTree.TopPanel({
-				dataProvider: TYPO3.Components.PageTree.DataProvider,
-				filteringTree: filteringTree,
-				ddGroup: this.id,
-				tree: tree,
-				app: this
-			});
-
-			var deletionDropZone = new TYPO3.Components.PageTree.DeletionDropZone({
-				id: this.id + '-deletionDropZone',
-				commandProvider: TYPO3.Components.PageTree.Actions,
-				ddGroup: this.id,
-				app: this,
-				region: 'south',
-				height: 35
-			});
-
-			var topPanelItems = new Ext.Panel({
-				id: this.id + '-topPanelItems',
-				border: false,
-				region: 'north',
-				height: 49,
-				items: [
-					topPanel, {
-						border: false,
-						id: this.id + '-indicatorBar'
-					}
-				]
-			});
-
-			this.add({
-				layout: 'border',
-				items: [
-					topPanelItems,
-					{
-						border: false,
-						id: this.id + '-treeContainer',
-						region: 'center',
-						layout: 'fit',
-						items: [tree, filteringTree]
-					},
-					deletionDropZone
-				]
-			});
-
-			if (TYPO3.Components.PageTree.Configuration.temporaryMountPoint) {
-				topPanelItems.on('afterrender', function() {
+		this.addListener('render', function () {
+			TYPO3.Components.PageTree.DataProvider.loadResources(function(response) {
+				TYPO3.Components.PageTree.LLL = response['LLL'];
+				TYPO3.Components.PageTree.Configuration = response['Configuration'];
+				TYPO3.Components.PageTree.Sprites = response['Sprites'];
+	
+				var tree = Ext.create('TYPO3.Components.PageTree.Tree', {
+					id: this.getId() + '-tree',
+					deletionDropZoneId: this.getId() + '-deletionDropZone',
+					ddGroup: this.getId(),
+					stateful: true,
+					stateId: 'Pagetree' + TYPO3.Components.PageTree.Configuration.temporaryMountPoint,
+					commandProvider: TYPO3.Components.PageTree.Actions,
+					contextMenuProvider: TYPO3.Components.PageTree.ContextMenuDataProvider,
+					treeDataProvider: TYPO3.Components.PageTree.DataProvider,
+					app: this
+				});
+	
+				var filteringTree = Ext.create('TYPO3.Components.PageTree.FilteringTree', {
+					id: this.getId() + '-filteringTree',
+					deletionDropZoneId: this.getId() + '-deletionDropZone',
+					ddGroup: this.getId(),
+					commandProvider: TYPO3.Components.PageTree.Actions,
+					contextMenuProvider: TYPO3.Components.PageTree.ContextMenuDataProvider,
+					stateful: false,
+					treeDataProvider: TYPO3.Components.PageTree.DataProvider,
+					app: this
+				});
+	
+				var topPanelItems = this.getComponent('topPanelItems');
+				topPanelItems.add([
+						Ext.create('TYPO3.Components.PageTree.TopPanel', {
+							border: false,
+							dataProvider: TYPO3.Components.PageTree.DataProvider,
+							filteringTree: filteringTree,
+							ddGroup: this.getId(),
+							height: 49,
+							itemId: 'topPanel',
+							tree: tree,
+							app: this
+						}),
+						{
+							xtype: 'container',
+							border: false,
+							defaultType: 'component',
+							id: this.getId() + '-indicatorBar',
+							itemId: 'indicatorBar'
+						}
+				]);
+	
+				var treeContainer = this.getComponent('treeContainer');
+				treeContainer.add([tree, filteringTree]);
+				treeContainer.addListener(
+					'render',
+					function (panel) { panel.getLayout().setActiveItem(0); }
+				);
+	
+				var deletionDropZoneContainer = this.getComponent('deletionDropZoneContainer');
+				deletionDropZoneContainer.add({
+					xtype: 'typo3deletiondropzone',
+					height: 35,
+					id: this.getId() + '-deletionDropZone',
+					commandProvider: TYPO3.Components.PageTree.Actions,
+					ddGroup: this.getId(),
+					layout: 'anchor',
+					app: this
+				});
+
+				if (TYPO3.Components.PageTree.Configuration.temporaryMountPoint) {
 					this.addTemporaryMountPointIndicator();
-				}, this);
-			}
-
-			if (TYPO3.Components.PageTree.Configuration.indicator !== '') {
-				this.addIndicatorItems();
-			}
-			this.doLayout();
-
-			this.ownerCt.on('resize', function() {
+				}
+	
+				if (TYPO3.Components.PageTree.Configuration.indicator !== '') {
+					this.addIndicatorItems();
+				}
 				this.doLayout();
-			});
+	
+				this.ownerCt.on('resize', function() {
+					this.doLayout();
+				}, this);
+			}, this);
 		}, this);
-
-		TYPO3.Components.PageTree.App.superclass.initComponent.apply(this, arguments);
+		this.callParent();
 	},
 
 	/**
@@ -187,9 +188,8 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 */
 	addIndicatorItems: function() {
 		this.addIndicator({
-			border: false,
-			id: this.id + '-indicatorBar-indicatorTitle',
-			cls: this.id + '-indicatorBar-item',
+			id: this.getId() + '-indicatorBar-indicatorTitle',
+			cls: this.getId() + '-indicatorBar-item',
 			html: TYPO3.Components.PageTree.Configuration.indicator
 		});
 	},
@@ -201,14 +201,13 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 */
 	addTemporaryMountPointIndicator: function() {
 		this.temporaryMountPointInfoIndicator = this.addIndicator({
-			border: false,
-			id: this.id + '-indicatorBar-temporaryMountPoint',
-			cls: this.id + '-indicatorBar-item',
+			id: this.getId() + '-indicatorBar-temporaryMountPoint',
+			cls: this.getId() + '-indicatorBar-item',
 
 			listeners: {
 				afterrender: {
 					fn: function() {
-						var element = Ext.fly(this.id + '-indicatorBar-temporaryMountPoint-clear');
+						var element = Ext.get(this.getId() + '-indicatorBar-temporaryMountPoint-clear');
 						element.on('click', function() {
 							TYPO3.BackendUserSettings.ExtDirect.unsetKey(
 								'pageTree_temporaryMountPoint',
@@ -225,18 +224,21 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 					scope: this
 				}
 			},
-
-			html: '<p>' +
-					'<span id="' + this.id + '-indicatorBar-temporaryMountPoint-info' + '" ' +
-						'class="' + this.id + '-indicatorBar-item-leftIcon ' +
-							TYPO3.Components.PageTree.Sprites.Info + '">&nbsp;' +
-					'</span>' +
-					'<span id="' + this.id + '-indicatorBar-temporaryMountPoint-clear' + '" ' +
-						'class="' + this.id + '-indicatorBar-item-rightIcon ' + '">X' +
-					'</span>' +
-					TYPO3.Components.PageTree.LLL.temporaryMountPointIndicatorInfo + '<br />' +
-						TYPO3.Components.PageTree.Configuration.temporaryMountPoint +
+			renderData: {
+				appId: this.getId(),
+				spriteIconCls: TYPO3.Components.PageTree.Sprites.Info,
+				label: TYPO3.Components.PageTree.LLL.temporaryMountPointIndicatorInfo,
+				mountPoint: TYPO3.Components.PageTree.Configuration.temporaryMountPoint
+			},
+			renderTpl: Ext.create('Ext.XTemplate',
+				'<p>',
+				'<span id="{appId}-indicatorBar-temporaryMountPoint-info" class="{appId}-indicatorBar-item-leftIcon {spriteIconCls}">&nbsp;</span>',
+				'{label}',
+				'<span id="{appId}-indicatorBar-temporaryMountPoint-clear" class="{appId}-indicatorBar-item-rightIcon t3-icon t3-icon-actions t3-icon-actions-input t3-icon-input-clear t3-tceforms-input-clearer">&nbsp;</span>',
+				'</p>',
+				'<p>{mountPoint}',
 				'</p>'
+			)
 		});
 	},
 
@@ -248,23 +250,18 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 */
 	addIndicator: function(component) {
 		if (component.listeners && component.listeners.afterrender) {
-			component.listeners.afterrender.fn = component.listeners.afterrender.fn.createSequence(
-				this.afterTopPanelItemAdded, this
-			);
+			component.listeners.afterrender.fn = Ext.Function.createSequence(component.listeners.afterrender.fn, this.afterTopPanelItemAdded, this);
 		} else {
-			if (component.listeners) {
-				component.listeners = {}
+			if (!component.listeners) {
+				component.listeners = {};
 			}
-
 			component.listeners.afterrender = {
 				scope: this,
 				fn: this.afterTopPanelItemAdded
 			}
 		}
-
-		var indicator = Ext.getCmp(this.id + '-indicatorBar').add(component);
-		this.doLayout();
-
+		var indicatorBar = this.down('component[itemId=indicatorBar]');
+		var indicator = indicatorBar.add(component);
 		return indicator;
 	},
 
@@ -275,8 +272,15 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	afterTopPanelItemAdded: function(component) {
-		var topPanelItems = Ext.getCmp(this.id + '-topPanelItems');
-		topPanelItems.setHeight(topPanelItems.getHeight() + component.getHeight() + 3);
+		var topPanel = this.down('component[itemId=topPanel]');
+		var indicatorBar = this.down('component[itemId=indicatorBar]');
+		var height = 0;
+		indicatorBar.items.each(function (item) {
+			height += item.getHeight() + 3;
+		});
+		indicatorBar.setHeight(height);
+		topPanelItems = this.down('component[itemId=topPanelItems]');
+		topPanelItems.setHeight(topPanel.getHeight() + indicatorBar.getHeight());
 		this.doLayout();
 	},
 
@@ -287,9 +291,9 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	removeIndicator: function(component) {
-		var topPanelItems = Ext.getCmp(this.id + '-topPanelItems');
+		var topPanelItems = this.down('component[itemId=topPanelItems]');
 		topPanelItems.setHeight(topPanelItems.getHeight() - component.getHeight() - 3);
-		Ext.getCmp(this.id + '-indicatorBar').remove(component);
+		Ext.getCmp(this.getId() + '-indicatorBar').remove(component);
 		this.doLayout();
 	},
 
@@ -318,11 +322,11 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 */
 	refreshTree: function() {
 		if (!isNaN(fsMod.recentIds['web']) && fsMod.recentIds['web'] !== '') {
-			this.select(fsMod.recentIds['web'], true);
+			this.select(fsMod.recentIds['web']);
 		}
 
 		TYPO3.Components.PageTree.DataProvider.getIndicators(function(response) {
-			var indicators = Ext.getCmp(this.id + '-indicatorBar-indicatorTitle');
+			var indicators = Ext.getCmp(this.getId() + '-indicatorBar-indicatorTitle');
 			if (indicators) {
 				this.removeIndicator(indicators);
 			}
@@ -333,7 +337,7 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 			}
 		}, this);
 
-		this.activeTree.refreshTree();
+		this.getTree().refreshTree();
 	},
 
 	/**
@@ -342,43 +346,44 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
 	 * @return {TYPO3.Components.PageTree.Tree}
 	 */
 	getTree: function() {
-		return this.activeTree;
+		return this.down('component[itemId=treeContainer]').getLayout().getActiveItem();
 	},
 
 	/**
-	 * Selects a node defined by the page id. If the second parameter is set, we
-	 * store the new location into the state hash.
+	 * Sets the current active tree
+	 * @param {TYPO3.Components.PageTree.Tree} tree
+	 * @return {TYPO3.Components.PageTree.Tree}
+	 */
+	setTree: function(tree) {
+		var layout = this.down('component[itemId=treeContainer]').getLayout();
+		layout.setActiveItem(tree);
+		return layout.getActiveItem(tree);
+	},
+
+	/**
+	 * Selects a node defined by the page id.
 	 *
 	 * @param {int} pageId
-	 * @param {Boolean} saveState
 	 * @return {Boolean}
 	 */
-	select: function(pageId, saveState) {
-		if (saveState !== false) {
-			saveState = true;
-		}
-
+	select: function(pageId) {
 		var tree = this.getTree();
 		var succeeded = false;
 		var node = tree.getRootNode().findChild('realId', pageId, true);
 		if (node) {
 			succeeded = true;
 			tree.selectPath(node.getPath());
-			if (!!saveState && tree.stateHash) {
-				tree.stateHash.lastSelectedNode = node.id;
-			}
 		}
-
 		return succeeded;
 	},
 
 	/**
 	 * Returns the currently selected node
 	 *
-	 * @return {Ext.tree.TreeNode}
+	 * @return {Ext.data.NodeInterface}
 	 */
 	getSelected: function() {
-		var node = this.getTree().getSelectionModel().getSelectedNode();
+		var node = this.getTree().getSelectionModel().getLastSelected();
 		return node ? node : null;
 	}
 });
@@ -389,15 +394,12 @@ TYPO3.Components.PageTree.App = Ext.extend(Ext.Panel, {
  * @return {TYPO3.Components.PageTree.App}
  */
 TYPO3.ModuleMenu.App.registerNavigationComponent('typo3-pagetree', function() {
-	TYPO3.Backend.NavigationContainer.PageTree = new TYPO3.Components.PageTree.App();
+	TYPO3.Backend.NavigationContainer.PageTree = Ext.create('TYPO3.Components.PageTree.App');
 
 		// compatibility code
-    top.nav = TYPO3.Backend.NavigationContainer.PageTree;
-    top.nav_frame = TYPO3.Backend.NavigationContainer.PageTree;
-    top.content.nav_frame = TYPO3.Backend.NavigationContainer.PageTree;
+	top.nav = TYPO3.Backend.NavigationContainer.PageTree;
+	top.nav_frame = TYPO3.Backend.NavigationContainer.PageTree;
+	top.content.nav_frame = TYPO3.Backend.NavigationContainer.PageTree;
 
 	return TYPO3.Backend.NavigationContainer.PageTree;
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.App', TYPO3.Components.PageTree.App);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/contextmenu.js b/t3lib/js/extjs/components/pagetree/javascript/contextmenu.js
index 32501b69f334eb1d66e489b917a9bb938ce10d1a..5c6ed687aa82b4189688e60b6cd0ac1c8e7b9269 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/contextmenu.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/contextmenu.js
@@ -23,8 +23,6 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.ContextMenu
  *
@@ -34,11 +32,12 @@ Ext.namespace('TYPO3.Components.PageTree');
  * @extends Ext.menu.Menu
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
+Ext.define('TYPO3.Components.PageTree.ContextMenu', {
+	extend: 'Ext.menu.Menu',
 	/**
 	 * Context menu node
 	 *
-	 * @cfg {Ext.tree.TreeNode}
+	 * @cfg {TYPO3.Components.PageTree.Model}
 	 */
 	node: null,
 
@@ -66,15 +65,15 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 	/**
 	 * Listeners
 	 *
-	 * The itemclick event triggers the configured single click action
+	 * The click event triggers the configured single click action
 	 */
 	listeners: {
-		itemclick: {
-			fn: function (item) {
-				if (this.pageTree.commandProvider[item.callbackAction]) {
+		click: {
+			fn: function (menu, item) {
+				if (item && this.pageTree.commandProvider[item.callbackAction]) {
 					if (item.parentMenu.pageTree.stateHash) {
-						fsMod.recentIds['web'] = item.parentMenu.node.attributes.nodeData.id;
-						item.parentMenu.pageTree.stateHash['lastSelectedNode'] = item.parentMenu.node.id;
+						fsMod.recentIds['web'] = item.parentMenu.node.getNodeData('id');
+						item.parentMenu.pageTree.stateHash['lastSelectedNode'] = item.parentMenu.node.getId();
 					}
 
 					this.pageTree.commandProvider[item.callbackAction](
@@ -90,12 +89,12 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 	/**
 	 * Fills the menu with the actions
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {TYPO3.Components.PageTree.Tree} pageTree
 	 * @param {Object} contextMenuConfiguration
 	 * @return {void}
 	 */
-	fill: function(node, pageTree, contextMenuConfiguration) {
+	fill: function (node, pageTree, contextMenuConfiguration) {
 		this.node = node;
 		this.pageTree = pageTree;
 
@@ -103,9 +102,11 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 		if (components.length) {
 			for (var component in components) {
 				if (components[component] === '-') {
-					this.addSeparator();
+					this.add({
+						xtype: 'menuseparator'
+					});
 				} else if (typeof components[component] === 'object') {
-					this.addItem(components[component]);
+					this.add(components[component]);
 				}
 			}
 		}
@@ -119,7 +120,7 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 	 * @param {int} level
 	 * @return {Object}
 	 */
-	preProcessContextMenuConfiguration: function(contextMenuConfiguration, level) {
+	preProcessContextMenuConfiguration: function (contextMenuConfiguration, level) {
 		level = level || 0;
 		if (level > 5) {
 			return [];
@@ -138,7 +139,7 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 				);
 
 				if (subMenuComponents.length) {
-					var subMenu = new TYPO3.Components.PageTree.ContextMenu({
+					var subMenu = Ext.create('TYPO3.Components.PageTree.ContextMenu', {
 						id: this.id + '-sub' + ++subMenus,
 						items: subMenuComponents,
 						node: this.node,
@@ -150,7 +151,8 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 						cls: 'contextMenu-subMenu',
 						menu: subMenu,
 						icon: contextMenuConfiguration[singleAction]['icon'],
-						iconCls: contextMenuConfiguration[singleAction]['class']
+						iconCls: contextMenuConfiguration[singleAction]['class'],
+						xtype: 'menuitem'
 					};
 				}
 			} else if (contextMenuConfiguration[singleAction]['type'] === 'divider') {
@@ -163,22 +165,14 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 
 				if (typeof contextMenuConfiguration[singleAction] === 'object') {
 					var component = {
-						'text': contextMenuConfiguration[singleAction]['label'],
-						'icon': contextMenuConfiguration[singleAction]['icon'],
-						'iconCls': contextMenuConfiguration[singleAction]['class'],
-						'callbackAction': contextMenuConfiguration[singleAction]['callbackAction'],
-						'customAttributes': contextMenuConfiguration[singleAction]['customAttributes']
+						text: contextMenuConfiguration[singleAction]['label'],
+						icon: contextMenuConfiguration[singleAction]['icon'],
+						iconCls: contextMenuConfiguration[singleAction]['class'],
+						callbackAction: contextMenuConfiguration[singleAction]['callbackAction'],
+						customAttributes: contextMenuConfiguration[singleAction]['customAttributes'],
+						xtype: 'menuitem'
 					};
 
-					component.itemTpl = Ext.menu.Item.prototype.itemTpl = new Ext.XTemplate(
-						'<a id="{id}" class="{cls}" hidefocus="true" unselectable="on" href="{href}">',
-							'<span class="{hrefTarget}">',
-								'<img src="{icon}" class="x-menu-item-icon {iconCls}" unselectable="on" />',
-							'</span>',
-							'<span class="x-menu-item-text">{text}</span>',
-						'</a>'
-					);
-
 					components[index++] = component;
 				}
 			}
@@ -192,6 +186,3 @@ TYPO3.Components.PageTree.ContextMenu = Ext.extend(Ext.menu.Menu, {
 		return components;
 	}
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.ContextMenu', TYPO3.Components.PageTree.ContextMenu);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/deletiondropzone.js b/t3lib/js/extjs/components/pagetree/javascript/deletiondropzone.js
index 443f96fa65880019049a6dbed5ef246c31e75262..a89afdbff79411f48580e4edf3519f14cf3410e1 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/deletiondropzone.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/deletiondropzone.js
@@ -23,18 +23,19 @@
  *
  *  This copyright notice MUST APPEAR in all copies of the script!
  ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.DeletionDropZone
  *
  * Deletion Drop Zone
  *
  * @namespace TYPO3.Components.PageTree
- * @extends Ext.Panel
+ * @extends Ext.panel.Panel
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
+Ext.define('TYPO3.Components.PageTree.DeletionDropZone', {
+	extend: 'Ext.panel.Panel',
+	alias: 'widget.typo3deletiondropzone',
+
 	/**
 	 * Border
 	 *
@@ -110,10 +111,7 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 
 				this.getEl().on('mouseout', function(e) {
 					if (!e.within(this.getEl(), true)) {
-						this.removeClass(this.id + '-activateProxyOver');
-						if (!this.app.activeTree.shouldCopyNode) {
-							this.app.activeTree.copyHint.show();
-						}
+						this.removeCls(this.id + '-activateProxyOver');
 					}
 				}, this);
 			}
@@ -139,7 +137,7 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 			'">&nbsp;</span><span id="' + this.id + '-text">' +
 			TYPO3.Components.PageTree.LLL.dropToRemove + '</span></p>';
 
-		TYPO3.Components.PageTree.DeletionDropZone.superclass.initComponent.apply(this, arguments);
+		this.callParent(arguments);
 	},
 
 
@@ -149,44 +147,49 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	createDropZone: function() {
-		(new Ext.dd.DropZone(this.getEl(), {
+		Ext.create('Ext.dd.DropZone', this.getEl(), {
 			ddGroup: this.ddGroup,
 
-			notifyOver: function(ddProxy, e) {
-				ddProxy.setDragElPos(e.xy[0], e.xy[1] - 60);
-				return this.id + '-proxyOver';
-			}.createDelegate(this),
-
-			notifyEnter: function() {
-				this.addClass(this.id + '-activateProxyOver');
-				if (!this.app.activeTree.shouldCopyNode) {
-					this.app.activeTree.copyHint.hide();
-				}
-
-				return this.id + '-proxyOver';
-			}.createDelegate(this),
-
-			notifyDrop: function(ddProxy, e, n) {
-				var node = n.node;
-				if (!node) {
-					return;
-				}
-
-				var tree = node.ownerTree;
-				var nodeHasChildNodes = (node.hasChildNodes() || node.isExpandable());
-
-				var callback = null;
-				if (!top.TYPO3.configuration.inWorkspace && !nodeHasChildNodes) {
-					callback = this.setRecoverState.createDelegate(this);
-				}
-
-				if (nodeHasChildNodes) {
-					node.ownerTree.commandProvider.confirmDelete(node, tree, callback, true);
-				} else {
-					node.ownerTree.commandProvider.deleteNode(node, tree, callback);
-				}
-			}.createDelegate(this)
-		}));
+			notifyOver: Ext.Function.bind(
+				function(ddProxy, e) {
+					ddProxy.setDragElPos(e.xy[0], e.xy[1] - 60);
+					return this.id + '-proxyOver';
+				},
+				this
+			),
+
+			notifyEnter: Ext.Function.bind(
+				function() {
+					this.addCls(this.id + '-activateProxyOver');
+					return this.id + '-proxyOver';
+				},
+				this
+			),
+
+			notifyDrop: Ext.Function.bind(
+				function (ddProxy, e, dragData) {
+					var view = dragData.view,
+						tree = view.panel,
+						node = view.getRecord(dragData.item);
+					if (!node) {
+						return;
+					}
+					var nodeHasChildNodes = (node.hasChildNodes() || node.isExpandable());
+	
+					var callback = null;
+					if (!top.TYPO3.configuration.inWorkspace && !nodeHasChildNodes) {
+						callback = Ext.Function.bind(this.setRecoverState, this);
+					}
+	
+					if (nodeHasChildNodes) {
+						tree.commandProvider.confirmDelete(node, tree, callback, true);
+					} else {
+						tree.commandProvider.deleteNode(node, tree, callback);
+					}
+				},
+				this
+			)
+		});
 	},
 
 	/**
@@ -207,7 +210,7 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 		this.setHeight(50);
 		this.updateIcon(TYPO3.Components.PageTree.Sprites.TrashCanRestore);
 		this.updateText(
-			node.text + '<br />' +
+			node.get('text') + '<br />' +
 			'<span class="' + this.id + '-restore">' +
 				'<span class="' + this.id + '-restoreText">' +
 				TYPO3.Components.PageTree.LLL.dropZoneElementRemoved +
@@ -218,13 +221,17 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 		this.app.doLayout();
 
 		++this.amountOfDrops;
-		(function() {
-			if (!--this.amountOfDrops) {
-				this.toOriginState();
-			}
-		}).defer(10000, this);
+		Ext.Function.defer(
+			function() {
+				if (!--this.amountOfDrops) {
+					this.toOriginState();
+				}
+			},
+			10000,
+			this
+		);
 
-		this.textClickHandler = this.restoreNode.createDelegate(this, [node, tree]);
+		this.textClickHandler = Ext.Function.bind(this.restoreNode, this, [node, tree]);
 		Ext.get(this.id + '-text').on('click', this.textClickHandler);
 
 		this.isPreviousSibling = false;
@@ -283,7 +290,7 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 		this.previousNode = this.textClickHandler = null;
 		this.isPreviousSibling = false;
 
-		if (hide && !this.app.activeTree.dragZone.dragging) {
+		if (hide && !this.app.getTree().getView().getPlugin('treeViewDragDrop').dragZone.dragging) {
 			this.hide();
 		}
 
@@ -310,13 +317,14 @@ TYPO3.Components.PageTree.DeletionDropZone = Ext.extend(Ext.Panel, {
 		this.updateText(TYPO3.Components.PageTree.LLL.dropZoneElementRestored);
 		this.app.doLayout();
 
-		(function() {
-			if (this.textClickHandler) {
-				this.toOriginState();
-			}
-		}).defer(3000, this);
+		Ext.Function.defer(
+			function() {
+				if (this.textClickHandler) {
+					this.toOriginState();
+				}
+			},
+			3000,
+			this
+		);
 	}
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.DeletionDropZone', TYPO3.Components.PageTree.DeletionDropZone);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/filteringtree.js b/t3lib/js/extjs/components/pagetree/javascript/filteringtree.js
index 7ee0d88d312de0316141e51525b4f52b217fcf51..5b24822d19bed59cf4281169e3883c513e9f8afa 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/filteringtree.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/filteringtree.js
@@ -23,8 +23,6 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.FilteringTree
  *
@@ -34,7 +32,9 @@ Ext.namespace('TYPO3.Components.PageTree');
  * @extends TYPO3.Components.PageTree.Tree
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.FilteringTree = Ext.extend(TYPO3.Components.PageTree.Tree, {
+Ext.define('TYPO3.Components.PageTree.FilteringTree', {
+	extend: 'TYPO3.Components.PageTree.Tree',
+
 	/**
 	 * Search word
 	 *
@@ -47,29 +47,45 @@ TYPO3.Components.PageTree.FilteringTree = Ext.extend(TYPO3.Components.PageTree.T
 	 *
 	 * @return {void}
 	 */
-	addTreeLoader: function() {
-		this.loader = new Ext.tree.TreeLoader({
-			directFn: this.treeDataProvider.getFilteredTree,
-			paramOrder: 'nodeId,attributes,searchWord',
-			nodeParameter: 'nodeId',
-			baseAttrs: {
-				uiProvider: this.uiProvider
-			},
-
-			listeners: {
-				beforeload: function(treeLoader, node) {
-					if (!node.ownerTree.searchWord || node.ownerTree.searchWord === '') {
-						return false;
+	addStore: function (store) {
+		this.store = Ext.data.StoreManager.lookup(this.getId() + 'FilteredPageTreeStore');
+		if (!this.store) {
+			this.store = Ext.create('Ext.data.TreeStore', {
+				clearOnLoad: false,
+				listeners: {
+					beforeload: {
+						fn: function (store, operation) {
+							if (!this.searchWord || this.searchWord === '') {
+								return false;
+							}
+							if (operation.node) {
+								var node = operation.node;
+								node.removeAll();
+								node.commit();
+								operation.params = {
+									nodeId: node.getNodeData('id'),
+									nodeData: node.get('nodeData'),
+									searchWord: this.searchWord
+									
+								};
+							}
+						},
+						scope: this
 					}
-
-					treeLoader.baseParams.nodeId = node.id;
-					treeLoader.baseParams.searchWord = node.ownerTree.searchWord;
-					treeLoader.baseParams.attributes = node.attributes.nodeData;
-				}
-			}
-		});
+				},
+				model: 'TYPO3.Components.PageTree.Model',
+				nodeParam: 'nodeId',
+				proxy: {
+					type: 'direct',
+					paramOrder: ['nodeId', 'nodeData', 'searchWord'],
+					directFn: this.treeDataProvider.getFilteredTree,
+					reader: {
+					    type: 'json'
+					}
+				},
+				root: this.rootNodeConfig,
+				storeId: this.getId() + 'FilteredPageTreeStore'
+			});
+		}
 	}
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.FilteringTree', TYPO3.Components.PageTree.FilteringTree);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/loadorder.txt b/t3lib/js/extjs/components/pagetree/javascript/loadorder.txt
index a649b22150ae39b41101fde397cc2c58bcdbd83b..84566aec2fd71794cf8fc9d2331d9d27d8aefb66 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/loadorder.txt
+++ b/t3lib/js/extjs/components/pagetree/javascript/loadorder.txt
@@ -1,10 +1,12 @@
 treeeditor.js
+pagetreecolumn.js
+contextmenu.js
+Ext.ux.state.TreePanel.js
+treedropzone.js
+treeviewdragdrop.js
 tree.js
 filteringtree.js
-nodeui.js
-deletiondropzone.js
 toppanel.js
-contextmenu.js
+deletiondropzone.js
 actions.js
-Ext.ux.state.TreePanel.js
 app.js
diff --git a/t3lib/js/extjs/components/pagetree/javascript/nodeui.js b/t3lib/js/extjs/components/pagetree/javascript/nodeui.js
deleted file mode 100644
index 1a1066063bcecd47c5a5add71a3c12506068feb4..0000000000000000000000000000000000000000
--- a/t3lib/js/extjs/components/pagetree/javascript/nodeui.js
+++ /dev/null
@@ -1,169 +0,0 @@
-/***************************************************************
-*  Copyright notice
-*
-*  (c) 2010-2011 Stefan Galinski <stefan.galinski@gmail.com>
-*  All rights reserved
-*
-*  This script is part of the TYPO3 project. The TYPO3 project is
-*  free software; you can redistribute it and/or modify
-*  it under the terms of the GNU General Public License as published by
-*  the Free Software Foundation; either version 2 of the License, or
-*  (at your option) any later version.
-*
-*  The GNU General Public License can be found at
-*  http://www.gnu.org/copyleft/gpl.html.
-*  A copy is found in the textfile GPL.txt and important notices to the license
-*  from the author is found in LICENSE.txt distributed with these scripts.
-*
-*
-*  This script is distributed in the hope that it will be useful,
-*  but WITHOUT ANY WARRANTY; without even the implied warranty of
-*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-*  GNU General Public License for more details.
-*
-*  This copyright notice MUST APPEAR in all copies of the script!
-***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
-/**
- * @class TYPO3.Components.PageTree.DeletionDropZone
- *
- * Tree Node User Interface that can handle sprite icons and more
- *
- * @namespace TYPO3.Components.PageTree
- * @extends Ext.tree.TreeNodeUI
- * @author Stefan Galinski <stefan.galinski@gmail.com>
- */
-TYPO3.Components.PageTree.PageTreeNodeUI = Ext.extend(Ext.tree.TreeNodeUI, {
-	/**
-	 * Adds the sprite icon and adds an event to open the context menu on a single click at the icon node
-	 *
-	 * @param {Ext.tree.TreeNode} n
-	 * @param {Object} a
-	 * @param {Ext.tree.TreeNode} targetNode
-	 * @param {Boolean} bulkRender
-	 * @return {void}
-	 */
-	renderElements : function(n, a, targetNode, bulkRender) {
-		// add some indent caching, this helps performance when rendering a large tree
-        this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
-
-        var cb = Ext.isBoolean(a.checked),
-            nel,
-            href = this.getHref(a.href),
-			rootline = '';
-
-			// TYPO3 modification to show the readable rootline above the user mounts
-		if (a.readableRootline !== '') {
-			var rootline = '<li class="x-tree-node-readableRootline">' + a.readableRootline + '</li>';
-		}
-
-		var buf = [rootline,'<li class="x-tree-node"><div ext:tree-node-id="',n.id,'" class="x-tree-node-el x-tree-node-leaf x-unselectable ', a.cls,'" unselectable="on">',
-            '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
-            '<img alt="" src="', this.emptyIcon, '" class="x-tree-ec-icon x-tree-elbow" />',
-//            '<img alt="" src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
-			a.spriteIconCode, // TYPO3: add sprite icon code
-            cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : '/>')) : '',
-            '<a hidefocus="on" class="x-tree-node-anchor" href="',href,'" tabIndex="1" ',
-             a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", '><span unselectable="on">',n.text,"</span></a></div>",
-            '<ul class="x-tree-node-ct" style="display:none;"></ul>',
-            "</li>"].join('');
-
-        if(bulkRender !== true && n.nextSibling && (nel = n.nextSibling.ui.getEl())){
-            this.wrap = Ext.DomHelper.insertHtml("beforeBegin", nel, buf);
-        }else{
-            this.wrap = Ext.DomHelper.insertHtml("beforeEnd", targetNode, buf);
-        }
-
-
-        this.elNode = this.wrap.childNodes[0];
-        this.ctNode = this.wrap.childNodes[1];
-        var cs = this.elNode.childNodes;
-        this.indentNode = cs[0];
-        this.ecNode = cs[1];
-//        this.iconNode = cs[2];
-		this.iconNode = (cs[2].firstChild.tagName === 'SPAN' ? cs[2].firstChild : cs[2]); // TYPO3: get possible overlay icon
-        var index = 3; // TYPO3: index 4?
-        if(cb){
-            this.checkbox = cs[3];
-            // fix for IE6
-            this.checkbox.defaultChecked = this.checkbox.checked;
-            index++;
-        }
-        this.anchor = cs[index];
-        this.textNode = cs[index].firstChild;
-
-			// TYPO3: call the context menu on a single click (Beware of drag&drop!)
-		if (!TYPO3.Components.PageTree.Configuration.disableIconLinkToContextmenu
-			|| TYPO3.Components.PageTree.Configuration.disableIconLinkToContextmenu === '0'
-		) {
-			Ext.fly(this.iconNode).on('click', function(event) {
-				this.getOwnerTree().fireEvent('contextmenu', this, event);
-				event.stopEvent();
-			}, n);
-		}
-	},
-
-	/**
-	 * Adds a quick tip to the sprite icon
-	 *
-	 * @param {Ext.tree.TreeNode} node
-	 * @param {Object} tip
-	 * @param {String} title
-	 * @return {void}
-	 */
-    onTipChange : function(node, tip, title) {
-		TYPO3.Components.PageTree.PageTreeNodeUI.superclass.onTipChange.apply(this, arguments);
-
-        if(this.rendered){
-            var hasTitle = Ext.isDefined(title);
-            if(this.iconNode.setAttributeNS){
-                this.iconNode.setAttributeNS("ext", "qtip", tip);
-                if(hasTitle){
-                    this.iconNode.setAttributeNS("ext", "qtitle", title);
-                }
-            }else{
-                this.iconNode.setAttribute("ext:qtip", tip);
-                if(hasTitle){
-                    this.iconNode.setAttribute("ext:qtitle", title);
-                }
-            }
-        }
-    },
-
-	/**
-	 * Returns the drag and drop handles
-	 *
-	 * @return {Object}
-	 */
-	getDDHandles: function() {
-		var ddHandles = [this.iconNode, this.textNode, this.elNode];
-		var handlesIndex = ddHandles.length;
-		
-		var textNode = Ext.get(this.textNode);
-		for (var i = 0; i < textNode.dom.childNodes.length; ++i) {
-			if (textNode.dom.childNodes[i].nodeName === 'SPAN') {
-				ddHandles[handlesIndex++] = textNode.dom.childNodes[i];
-				if (textNode.dom.childNodes[i].childNodes) {
-					ddHandles[handlesIndex++] = textNode.dom.childNodes[i].childNodes[0];
-				}
-			}
-		}
-
-		return ddHandles;
-    },
-
-	/**
-	 * Only set the onOver class if we are not in dragging mode
-	 *
-	 * @return {void}
-	 */
-	onOver: function() {
-		if (!this.node.ownerTree.dontSetOverClass) {
-			TYPO3.Components.PageTree.PageTreeNodeUI.superclass.onOver.apply(this, arguments);
-		}
-	}
-});
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.PageTreeNodeUI', TYPO3.Components.PageTree.PageTreeNodeUI);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/pagetreecolumn.js b/t3lib/js/extjs/components/pagetree/javascript/pagetreecolumn.js
new file mode 100644
index 0000000000000000000000000000000000000000..e4941d313a5c622f28928a6e83ed1896c461927c
--- /dev/null
+++ b/t3lib/js/extjs/components/pagetree/javascript/pagetreecolumn.js
@@ -0,0 +1,166 @@
+/***************************************************************
+*  Copyright notice
+*
+*  (c) 2011 Stanislas Rolland <typo3@sjbr.ca>
+*  All rights reserved
+*
+*  This script is part of the TYPO3 project. The TYPO3 project is
+*  free software; you can redistribute it and/or modify
+*  it under the terms of the GNU General Public License as published by
+*  the Free Software Foundation; either version 2 of the License, or
+*  (at your option) any later version.
+*
+*  The GNU General Public License can be found at
+*  http://www.gnu.org/copyleft/gpl.html.
+*  A copy is found in the textfile GPL.txt and important notices to the license
+*  from the author is found in LICENSE.txt distributed with these scripts.
+*
+*
+*  This script is distributed in the hope that it will be useful,
+*  but WITHOUT ANY WARRANTY; without even the implied warranty of
+*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+*  GNU General Public License for more details.
+*
+*  This copyright notice MUST APPEAR in all copies of the script!
+***************************************************************/
+/**
+ * @class TYPO3.Components.PageTree.Column
+ *
+ * Page tree column
+ *
+ * @namespace TYPO3.Components.PageTree
+ * @extends Ext.grid.column.Column
+ * @author Stanislas Rolland <typo3@sjbr.ca>
+ */
+Ext.define('TYPO3.Components.PageTree.Column', {
+	extend: 'Ext.grid.column.Column',
+	alias: 'widget.pagetreecolumn',
+	
+	/**
+	 * Render a row of the page tree adding relevant icons depending on record depth and record type
+	 *
+	 */
+	initComponent: function() {
+		var origRenderer = this.renderer || this.defaultRenderer,
+			origScope = this.scope || window;
+
+		this.renderer = function(value, metaData, record, rowIdx, colIdx, store, view) {
+			var buf   = [],
+				format = Ext.String.format,
+				depth = record.getDepth(),
+				treePrefix = Ext.baseCSSPrefix + 'tree-',
+				elbowPrefix = treePrefix + 'elbow-',
+				expanderCls = treePrefix + 'expander',
+				imgText = '<img src="{1}" class="{0}" />',
+				checkboxText = '<input type="button" role="checkbox" class="{0}" {1} />',
+				formattedValue = origRenderer.apply(origScope, arguments),
+				checked = null,
+				href = record.getNodeData('href'),
+				target = record.getNodeData('hrefTarget'),
+				cls = record.getNodeData('cls'),
+				readableRootline = record.getNodeData('readableRootline');
+	
+			while (record) {
+				if (!record.isRoot() || (record.isRoot() && view.rootVisible)) {
+					if (record.getDepth() === depth) {
+							// Add TYPO3 sprite icon code
+						buf.unshift(record.getNodeData('spriteIconCode'));
+							// Check if nodeData['checked'] is boolean
+						checked = record.getNodeData('checked');
+						if (Ext.isBoolean(checked)) {
+							buf.unshift(format(
+								checkboxText,
+								(treePrefix + 'checkbox') + (checked ? ' ' + treePrefix + 'checkbox-checked' : ''),
+								checked ? 'aria-checked="true"' : ''
+							));
+							if (checked) {
+								metaData.tdCls += (' ' + treePrefix + 'checked');
+							}
+						}
+							// Remove +/arrow from the TYPO3 root
+						if (record.getDepth() === 1) {
+							buf.unshift(format(imgText, (elbowPrefix + 'empty'), Ext.BLANK_IMAGE_URL));
+						} else if (record.isLast()) {
+							if (record.isExpandable()) {
+								buf.unshift(format(imgText, (elbowPrefix + 'end-plus ' + expanderCls), Ext.BLANK_IMAGE_URL));
+							} else {
+								buf.unshift(format(imgText, (elbowPrefix + 'end'), Ext.BLANK_IMAGE_URL));
+							}
+						} else {
+							if (record.isExpandable()) {
+								buf.unshift(format(imgText, (elbowPrefix + 'plus ' + expanderCls), Ext.BLANK_IMAGE_URL));
+							} else {
+								buf.unshift(format(imgText, (treePrefix + 'elbow'), Ext.BLANK_IMAGE_URL));
+							}
+						}
+					} else {
+							// Remove elbow from the TYPO3 root
+						if (record.isLast() || record.getDepth() === 1) {
+							buf.unshift(format(imgText, (elbowPrefix + 'empty'), Ext.BLANK_IMAGE_URL));
+						} else if (record.getDepth() !== 1) {
+							buf.unshift(format(imgText, (elbowPrefix + 'line'), Ext.BLANK_IMAGE_URL));
+						}                      
+					}
+				}
+				record = record.parentNode;
+			} 
+			if (href) {
+				buf.push('<a href="', href, '" target="', target, '">', formattedValue, '</a>');
+			} else {
+				buf.push(formattedValue);
+			}
+			if (cls) {
+				metaData.tdCls += ' ' + cls;
+			}
+				// Show the readable rootline above the user mounts
+			if (readableRootline !== '') {
+				buf.unshift('<div class="x-tree-node-readableRootline">' + readableRootline + '</div>');
+			}
+			return buf.join('');
+		};
+		this.callParent(arguments);
+	},
+
+	defaultRenderer: function (value) {
+		return value;
+	},
+	/**
+	 * Create editing field with correct margin left
+	 *
+	 */
+        getEditor: function (record, defaultField) {
+		var depth = parseInt(record.getDepth()),
+			marginLeft = 51;
+		if (depth > 2) {
+			marginLeft += (depth-2)*16;
+		}
+		var field = this.field;
+        	if (!field && this.editor) {
+			field = this.editor;
+			delete this.editor;
+		}
+
+		if (!field && defaultField) {
+			field = defaultField;
+		}
+
+		if (field) {
+ 			if (Ext.isString(field)) {
+				field = { xtype: field };
+			}
+			if (field.isFormField) {
+				field.setFieldStyle({
+					marginLeft: marginLeft + 'px'
+				});
+			}
+			if (Ext.isObject(field) && !field.isFormField) {
+				field = Ext.ComponentManager.create(field, 'textfield');
+				this.field = field;
+			}
+			Ext.apply(field, {
+				name: this.dataIndex
+			});
+			return field;
+		}
+	}
+});
diff --git a/t3lib/js/extjs/components/pagetree/javascript/toppanel.js b/t3lib/js/extjs/components/pagetree/javascript/toppanel.js
index f80299d8e9a3ec0c7dede09757fb1b996b71d9b3..04d28f9749a4e34dc861e1f61198edfc21e0e74e 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/toppanel.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/toppanel.js
@@ -23,18 +23,18 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.TopPanel
  *
  * Top Panel
  *
  * @namespace TYPO3.Components.PageTree
- * @extends Ext.Panel
+ * @extends Ext.panel.Panel
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
+Ext.define('TYPO3.Components.PageTree.TopPanel', {
+	extend: 'Ext.panel.Panel',
+
 	/**
 	 * Component Id
 	 *
@@ -43,37 +43,48 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	id: 'typo3-pagetree-topPanel',
 
 	/**
-	 * Border
+	 * Toolbar Object
 	 *
-	 * @type {Boolean}
+	 * @type {Ext.Toolbar}
 	 */
-	border: false,
+	dockedItems: [{
+		xtype: 'toolbar',
+		dock: 'top',
+		itemId: 'topToolbar'
+	}],
 
 	/**
-	 * Toolbar Object
+	 * Panel CSS
 	 *
-	 * @type {Ext.Toolbar}
+	 * @type {String}
 	 */
-	tbar: new Ext.Toolbar(),
-
+	cls: 'typo3-pagetree-topPanel',
+	
 	/**
-	 * Currently Clicked Toolbar Button
+	 * Body CSS
 	 *
-	 * @type {Ext.Button}
+	 * @type {String}
 	 */
-	currentlyClickedButton: null,
+	bodyCls: 'typo3-pagetree-topPanel-item',
+	
+	/**
+	 * Layout
+	 *
+	 * @type {String}
+	 */
+	layout: 'anchor',
 
 	/**
-	 * Currently Shown Panel
+	 * Currently Clicked Toolbar Button
 	 *
-	 * @type {Ext.Component}
+	 * @type {Ext.Button}
 	 */
-	currentlyShownPanel: null,
+	currentlyClickedButton: null,
 
 	/**
 	 * Filtering Indicator Item
 	 *
-	 * @type {Ext.Panel}
+	 * @type {Ext.panel.Panel}
 	 */
 	filteringIndicator: null,
 
@@ -118,38 +129,20 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	initComponent: function() {
-		this.currentlyShownPanel = new Ext.Panel({
-			id: this.id + '-defaultPanel',
-			cls: this.id + '-item'
-		});
-		this.items = [this.currentlyShownPanel];
-
-		TYPO3.Components.PageTree.TopPanel.superclass.initComponent.apply(this, arguments);
-
+		this.callParent();
+			// Node insertion feature
 		this.addDragDropNodeInsertionFeature();
-
+			// Filter feature
 		if (!TYPO3.Components.PageTree.Configuration.hideFilter
 			|| TYPO3.Components.PageTree.Configuration.hideFilter === '0'
 		) {
 			this.addFilterFeature();
 		}
-
-		this.getTopToolbar().addItem({xtype: 'tbfill'});
+			// Refresh feature
+		this.getDockedComponent('topToolbar').add({xtype: 'tbfill'});
 		this.addRefreshTreeFeature();
 	},
 
-	/**
-	 * Returns a custom button template to fix some nasty webkit issues
-	 * by removing some useless wrapping html code
-	 *
-	 * @return {void}
-	 */
-	getButtonTemplate: function() {
-		return new Ext.Template(
-			'<div id="{4}" class="x-btn {3}"><button type="{0}"">&nbsp;</button></div>'
-		);
-	},
-
 	/**
 	 * Adds a button to the components toolbar with a related component
 	 *
@@ -158,7 +151,6 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	addButton: function(button, connectedWidget) {
-		button.template = this.getButtonTemplate();
 		if (!button.hasListener('click')) {
 			button.on('click', this.topbarButtonCallback);
 		}
@@ -169,7 +161,7 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 			this.add(connectedWidget);
 		}
 
-		this.getTopToolbar().addItem(button);
+		this.getDockedComponent('topToolbar').add(button);
 		this.doLayout();
 	},
 
@@ -182,44 +174,45 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	topbarButtonCallback: function() {
 		var topPanel = this.ownerCt.ownerCt;
 
-		topPanel.currentlyShownPanel.hide();
 		if (topPanel.currentlyClickedButton) {
 			topPanel.currentlyClickedButton.toggle(false);
+			if (topPanel.currentlyClickedButton.connectedWidget) {
+				topPanel.currentlyClickedButton.connectedWidget.hide();
+			}
 		}
 
 		if (topPanel.currentlyClickedButton === this) {
 			topPanel.currentlyClickedButton = null;
-			topPanel.currentlyShownPanel = topPanel.get(topPanel.id + '-defaultPanel');
+			if (this.connectedWidget) {
+				this.connectedWidget.hide();
+			}
 		} else {
 			this.toggle(true);
 			topPanel.currentlyClickedButton = this;
-			topPanel.currentlyShownPanel = this.connectedWidget;
+			if (this.connectedWidget) {
+				this.connectedWidget.show();
+			}
 		}
-
-		topPanel.currentlyShownPanel.show();
 	},
 
 	/**
 	 * Loads the filtering tree nodes with the given search word
 	 *
-	 * @param {Ext.form.TextField} textField
+	 * @param {Ext.form.field.Trigger} textField
 	 * @return {void}
 	 */
-	createFilterTree: function(textField) {
+	createFilterTree: function (textField) {
 		var searchWord = textField.getValue();
 		var isNumber = TYPO3.Utility.isNumber(searchWord);
 		var hasMinLength = (searchWord.length > 2 || searchWord.length <= 0);
 		if ((!hasMinLength && !isNumber) || searchWord === this.filteringTree.searchWord) {
 			return;
 		}
-
 		this.filteringTree.searchWord = searchWord;
 		if (this.filteringTree.searchWord === '') {
-			this.app.activeTree = this.tree;
-
 			textField.setHideTrigger(true);
-			this.filteringTree.hide();
-			this.tree.show().refreshTree(function() {
+			this.app.setTree(this.tree);
+			this.tree.refreshTree(function() {
 				textField.focus(false, 500);
 			}, this);
 
@@ -229,7 +222,6 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 			}
 		} else {
 			var selectedNode = this.app.getSelected();
-			this.app.activeTree = this.filteringTree;
 
 			if (!this.filteringIndicator) {
 				this.filteringIndicator = this.app.addIndicator(
@@ -237,20 +229,20 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 				);
 			}
 
-			textField.setHideTrigger(false);
-			this.tree.hide();
 			this.app.ownerCt.getEl().mask('', 'x-mask-loading-message');
-			this.app.ownerCt.getEl().addClass('t3-mask-loading');
-			this.filteringTree.show().refreshTree(function() {
+			this.app.ownerCt.getEl().addCls('t3-mask-loading');
+			this.app.setTree(this.filteringTree);
+			this.filteringTree.refreshTree(function() {
 				if (selectedNode) {
-					this.app.select(selectedNode.attributes.nodeData.id, false);
+					this.app.select(selectedNode.getNodeData('id'));
 				}
-				textField.focus();
 				this.app.ownerCt.getEl().unmask();
+				textField.setHideTrigger(false);
+				textField.triggerWrap.setWidth(0);
+				this.forceComponentLayout();
+				textField.focus();
 			}, this);
 		}
-
-		this.doLayout();
 	},
 
 	/**
@@ -261,26 +253,27 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 */
 	createIndicatorItem: function(textField) {
 		return {
-			border: false,
-			id: this.app.id + '-indicatorBar-filter',
-			cls: this.app.id + '-indicatorBar-item',
-			html: '<p>' +
-					'<span id="' + this.app.id + '-indicatorBar-filter-info' + '" ' +
-						'class="' + this.app.id + '-indicatorBar-item-leftIcon ' +
-							TYPO3.Components.PageTree.Sprites.Info + '">&nbsp;' +
-					'</span>' +
-					'<span id="' + this.app.id + '-indicatorBar-filter-clear' + '" ' +
-						'class="' + this.app.id + '-indicatorBar-item-rightIcon ' + '">X' +
-					'</span>' +
-					TYPO3.Components.PageTree.LLL.activeFilterMode +
-				'</p>',
+			id: this.app.getId() + '-indicatorBar-filter',
+			cls: this.app.getId() + '-indicatorBar-item',
+			renderData: {
+				appId: this.app.getId(),
+				spriteIconCls: TYPO3.Components.PageTree.Sprites.Info,
+				label: TYPO3.Components.PageTree.LLL.activeFilterMode
+			},
+			renderTpl: Ext.create('Ext.XTemplate',
+				'<p>',
+				'<span id="{appId}-indicatorBar-filter-info" class="{appId}-indicatorBar-item-leftIcon {spriteIconCls}">&nbsp;</span>',
+				'&nbsp;{label}&nbsp;',
+				'<span id="{appId}-indicatorBar-filter-clear" class="{appId}-indicatorBar-item-rightIcon">X</span>',
+				'</p>'
+			),
 			filteringTree: this.filteringTree,
 
 			listeners: {
 				afterrender: {
 					scope: this,
-					fn: function() {
-						var element = Ext.fly(this.app.id + '-indicatorBar-filter-clear');
+					fn: function(component) {
+						var element = Ext.get(this.app.getId() + '-indicatorBar-filter-clear');
 						element.on('click', function() {
 							textField.setValue('');
 							this.createFilterTree(textField);
@@ -297,71 +290,65 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	addFilterFeature: function() {
-		var topPanelButton = new Ext.Button({
-			id: this.id + '-button-filter',
-			cls: this.id + '-button',
+		var topPanelButton = Ext.create('Ext.button.Button', {
+			id: this.getId() + '-button-filter',
+			cls: this.getId() + '-button',
 			iconCls: TYPO3.Components.PageTree.Sprites.Filter,
 			tooltip: TYPO3.Components.PageTree.LLL.buttonFilter
 		});
 
-		var textField = new Ext.form.TriggerField({
-			id: this.id + '-filter',
+		var textField = Ext.create('Ext.form.field.Trigger', {
+			id: this.getId() + '-filter',
+			border: false,
 			enableKeyEvents: true,
-			triggerClass: TYPO3.Components.PageTree.Sprites.InputClear,
+			labelWidth: 0,
+			triggerCls: TYPO3.Components.PageTree.Sprites.InputClear,
 			value: TYPO3.Components.PageTree.LLL.searchTermInfo,
 
 			listeners: {
 				blur: {
-					scope: this,
-					fn:function(textField) {
+					fn: function (textField) {
 						if (textField.getValue() === '') {
 							textField.setValue(TYPO3.Components.PageTree.LLL.searchTermInfo);
-							textField.addClass(this.id + '-filter-defaultText');
+							textField.inputEl.addCls(this.getId() + '-filter-defaultText');
 						}
-					}
+					},
+					scope: this
 				},
 
 				focus: {
-					scope: this,
-					fn: function(textField) {
+					fn: function (textField) {
 						if (textField.getValue() === TYPO3.Components.PageTree.LLL.searchTermInfo) {
 							textField.setValue('');
-							textField.removeClass(this.id + '-filter-defaultText');
+							textField.inputEl.removeCls(this.getId() + '-filter-defaultText');
 						}
-					}
+					},
+					scope: this
 				},
 
 				keydown: {
 					fn: this.createFilterTree,
 					scope: this,
 					buffer: 1000
-				}
-			}
-		});
-
-		textField.setHideTrigger(true);
-		textField.onTriggerClick = function() {
-			textField.setValue('');
-			this.createFilterTree(textField);
-		}.createDelegate(this);
-
-		var topPanelWidget = new Ext.Panel({
-			border: false,
-			id: this.id + '-filterWrap',
-			cls: this.id + '-item',
-			items: [textField],
+				},
 
-			listeners: {
 				show: {
-					scope: this,
-					fn: function(panel) {
-						panel.get(this.id + '-filter').focus();
-					}
+					fn: function () { this.focus(); }
 				}
 			}
 		});
 
-		this.addButton(topPanelButton, topPanelWidget);
+		textField.setHideTrigger(true);
+		textField.onTriggerClick = Ext.Function.bind(
+			function (textField) {
+				textField.setValue('');
+				this.createFilterTree(textField);
+			},
+			this,
+			[textField]
+		);
+
+		this.addButton(topPanelButton, textField);
 	},
 
 	/**
@@ -370,28 +357,32 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	createNewNodeToolbar: function() {
-		this.dragZone = new Ext.dd.DragZone(this.getEl(), {
+		this.dragZone = Ext.create('Ext.dd.DragZone', this.getEl(), {
 			ddGroup: this.ownerCt.ddGroup,
 			topPanel: this.ownerCt,
 
 			endDrag: function() {
-				this.topPanel.app.activeTree.dontSetOverClass = false;
+				this.topPanel.app.getTree().dontSetOverClass = false;
 			},
 
-			getDragData: function(event) {
+			getDragData: function (event) {
 				this.proxyElement = document.createElement('div');
-
-				var node = Ext.getCmp(event.getTarget('.x-btn').id);
-				node.shouldCreateNewNode = true;
-
+				var clickedButton = event.getTarget('.x-btn');
+				if (clickedButton) {
+					var node = Ext.getCmp(clickedButton.id);
+				}
+				if (node) {
+					node.shouldCreateNewNode = true;
+				}
 				return {
 					ddel: this.proxyElement,
-					item: node
+					item: node,
+					records: [node]
 				}
 			},
 
 			onInitDrag: function() {
-				this.topPanel.app.activeTree.dontSetOverClass = true;
+				this.topPanel.app.getTree().dontSetOverClass = true;
 				var clickedButton = this.dragData.item;
 				var cls = clickedButton.initialConfig.iconCls;
 
@@ -405,18 +396,18 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 			}
 		});
 
-			// listens on the escape key to stop the dragging
-		(new Ext.KeyMap(document, {
+			// Listens on the escape key to stop the dragging
+		Ext.create('Ext.util.KeyMap', document, {
 			key: Ext.EventObject.ESC,
 			scope: this,
 			buffer: 250,
 			fn: function(event) {
 				if (this.dragZone.dragging) {
-					Ext.dd.DragDropMgr.stopDrag(event);
+					Ext.dd.DragDropManager.stopDrag(event);
 					this.dragZone.onInvalidDrop(event);
 				}
 			}
-		}, 'keydown'));
+		}, 'keydown');
 	},
 
 	/**
@@ -425,11 +416,16 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	addDragDropNodeInsertionFeature: function() {
-		var newNodeToolbar = new Ext.Toolbar({
-			border: false,
-			id: this.id + '-item-newNode',
-			cls: this.id + '-item',
+		var topPanelButton = Ext.create('Ext.button.Button', {
+			id: this.getId() + '-button-newNode',
+			cls: this.getId() + '-button',
+			iconCls: TYPO3.Components.PageTree.Sprites.NewNode,
+			tooltip: TYPO3.Components.PageTree.LLL.buttonNewNode
+		});
 
+		var newNodeToolbar = Ext.create('Ext.toolbar.Toolbar', {
+			id: this.getId() + '-item-newNode',
+			cls: this.getId() + '-item',
 			listeners: {
 				render: {
 					fn: this.createNewNodeToolbar
@@ -438,20 +434,10 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 		});
 
 		this.dataProvider.getNodeTypes(function(response) {
-			for (var i = 0; i < response.length; ++i) {
-				response[i].template = this.getButtonTemplate();
-				newNodeToolbar.addItem(response[i]);
-			}
+			newNodeToolbar.add(response);
 			newNodeToolbar.doLayout();
 		}, this);
 
-		var topPanelButton = new Ext.Button({
-			id: this.id + '-button-newNode',
-			cls: this.id + '-button',
-			iconCls: TYPO3.Components.PageTree.Sprites.NewNode,
-			tooltip: TYPO3.Components.PageTree.LLL.buttonNewNode
-		});
-
 		this.addButton(topPanelButton, newNodeToolbar);
 	},
 
@@ -461,9 +447,9 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 	 * @return {void}
 	 */
 	addRefreshTreeFeature: function() {
-		var topPanelButton = new Ext.Button({
-			id: this.id + '-button-refresh',
-			cls: this.id + '-button',
+		var topPanelButton = Ext.create('Ext.button.Button', {
+			id: this.getId() + '-button-refresh',
+			cls: this.getId() + '-button',
 			iconCls: TYPO3.Components.PageTree.Sprites.Refresh,
 			tooltip: TYPO3.Components.PageTree.LLL.buttonRefresh,
 
@@ -471,7 +457,7 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 				click: {
 					scope: this,
 					fn: function() {
-						this.app.activeTree.refreshTree();
+						this.app.getTree().refreshTree();
 					}
 				}
 			}
@@ -480,6 +466,3 @@ TYPO3.Components.PageTree.TopPanel = Ext.extend(Ext.Panel, {
 		this.addButton(topPanelButton);
 	}
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.TopPanel', TYPO3.Components.PageTree.TopPanel);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/tree.js b/t3lib/js/extjs/components/pagetree/javascript/tree.js
index 7319b02472c735c5b349395d1c92b273e4628d0b..fe61c6a8e946c05a169daaacea432fdec7604f6c 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/tree.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/tree.js
@@ -23,43 +23,145 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.Tree
  *
  * Generic Tree Panel
  *
  * @namespace TYPO3.Components.PageTree
- * @extends Ext.tree.TreePanel
+ * @extends Ext.tree.Panel
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
+Ext.define('TYPO3.Components.PageTree.Model', {
+	extend: 'Ext.data.Model',
+	fields: [{
+		name: 'id',
+		type: 'string',
+		defaultValue: 'root'
+	},{
+		name: 'realId',
+		type: 'string'
+	},{
+		name: 'text',
+		type: 'string'
+	},{
+		name: 'depth',
+		type: 'int'
+	},{
+		name: 'root',
+		type: 'boolean'
+	},{
+		name: 'leaf',
+		type: 'boolean'
+	},{
+		name: 'parentId',
+		type: 'string'
+	},{
+		name: 'isFirst',
+		type: 'boolean'
+	},{
+		name: 'index',
+		type: 'int'
+	},{
+		name: 'isLast',
+		type: 'boolean'
+	},{
+		name: 'isExpandable',
+		type: 'boolean'
+	},{
+		name: 'isInsertedNode',
+		type: 'boolean'
+	},{
+		name: 'nodeData',
+		type: 'object'
+	}],
+	hasMany: {
+		name: 'children',
+		associationKey: 'children',
+		model: 'TYPO3.Components.PageTree.Model'
+	},
+		// Set method for nodeData fields
+	setNodeData: function (field, value) {
+		var nodeData = this.get('nodeData');
+		nodeData[field] = value;
+		this.set('nodeData', Ext.merge(this.get('nodeData'), nodeData));
+	},
+		// Get method for nodeData fields
+	getNodeData: function (field) {
+		return this.get('nodeData')[field];
+	}
+});
+Ext.define('TYPO3.Components.PageTree.Tree', {
+	extend: 'Ext.tree.Panel',
+	/**
+	 * Use extended stateful mixin
+	 *
+	 * @type {Object}
+	 */
+	mixins: {
+		state: 'Ext.ux.state.TreePanel'
+	},
+
+	/**
+	 * View configuration
+	 *
+	 * @type {Object}
+	 */
+	viewConfig: {
+		autoScroll: false,
+		border: false,
+		toggleOnDblClick: false
+	},
+
+	/**
+	 * Columns
+	 *
+	 * @type {TYPO3.Components.PageTree.Column}[]
+	 */
+	columns: [{
+	 		xtype: 'pagetreecolumn',
+	 	 	dataIndex: 'text',
+	 	 	flex: 1,
+	 	 	editor: {
+	 	 		xtype: 'textfield',
+	 	 	 	allowBlank: false
+	 	 	}
+	}],
+
+	/**
+	 * Header
+	 *
+	 * @type {Boolean}
+	 */
+	hideHeaders: true,
+	preventHeader: true,
+
 	/**
 	 * Border
 	 *
 	 * @type {Boolean}
 	 */
+	autoScroll: false,
 	border: false,
 
 	/**
-	 * Indicates if the root node is visible
+	 * Body css
 	 *
-	 * @type {Boolean}
+	 * @type {String}
 	 */
-	rootVisible: false,
+	bodyCls: 'typo3-pagetree',
 
 	/**
-	 * Tree Editor Instance (Inline Edit)
+	 * Indicates if the root node is visible
 	 *
-	 * @type {TYPO3.Components.PageTree.TreeEditor}
+	 * @type {Boolean}
 	 */
-	treeEditor: null,
+	rootVisible: false,
 
 	/**
 	 * Currently Selected Node
 	 *
-	 * @type {Ext.tree.TreeNode}
+	 * @type {TYPO3.Components.PageTree.Model}
 	 */
 	currentSelectedNode: null,
 
@@ -78,18 +180,18 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	ddGroup: '',
 
 	/**
-	 * Indicates if the label should be editable
+	 * Id of deletionDropZone
 	 *
-	 * @cfg {Boolean}
+	 * @cfg {String}
 	 */
-	labelEdit: true,
+	deletionDropZoneId: '',
 
 	/**
-	 * User Interface Provider
+	 * Indicates if the label should be editable
 	 *
-	 * @cfg {Ext.tree.TreeNodeUI}
+	 * @cfg {Boolean}
 	 */
-	uiProvider: null,
+	labelEdit: true,
 
 	/**
 	 * Data Provider
@@ -103,7 +205,7 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	 *
 	 * @cfg {Object}
 	 */
-	commandProvider : null,
+	commandProvider: null,
 
 	/**
 	 * Context menu provider
@@ -113,18 +215,18 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	contextMenuProvider: null,
 
 	/**
-	 * Id of the deletion drop zone if any
+	 * Main applicaton
 	 *
-	 * @cfg {String}
+	 * @cfg {TYPO3.Components.PageTree.App}
 	 */
-	deletionDropZoneId: '',
+	app: null,
 
 	/**
-	 * Main applicaton
+	 * Page Tree Store
 	 *
-	 * @cfg {TYPO3.Components.PageTree.App}
+	 * @type {Object}
 	 */
-	app: null,
+	store: null,
 
 	/**
 	 * Root Node Configuration
@@ -139,17 +241,10 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 		}
 	},
 
-	/**
-	 * Indicator if the control key is pressed
-	 *
-	 * @type {Boolean}
-	 */
-	isControlPressed: false,
-
 	/**
 	 * Context Node
 	 *
-	 * @type {Ext.tree.TreeNode}
+	 * @type {TYPO3.Components.PageTree.Model}
 	 */
 	t3ContextNode: null,
 
@@ -158,68 +253,52 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	 *
 	 * @type {Object}
 	 */
-	t3ContextInfo: {
+	 t3ContextInfo: {
 		inCopyMode: false,
 		inCutMode: false
 	},
 
 	/**
-	 * Registered clicks for the double click feature
+	 * Number of clicks to ignore for the label edit on dblclick feature
+	 * Will be set to 2 by the tree editor
 	 *
 	 * @type {int}
 	 */
-	clicksRegistered: 0,
+	inhibitClicks: 0,
 
 	/**
-	 * Indicator if the control key was pressed
+	 * Constructor
+	 * Plugins are built by the parent constructor
 	 *
-	 * @type {Boolean}
-	 */
-	controlKeyPressed: false,
-
-	/**
-	 * Listeners
-	 *
-	 * Event handlers that handle click events and synchronizes the label edit,
-	 * double click and single click events in a useful way.
+	 * @param {Object} config
+	 * @return {void}
 	 */
-	listeners: {
-			// single click handler that only triggers after a delay to let the double click event
-			// a possibility to be executed (needed for label edit)
-		click: {
-			fn: function(node, event) {
-				if (this.clicksRegistered === 2) {
-					this.clicksRegistered = 0;
-					event.stopEvent();
-					return false;
-				}
-
-				this.clicksRegistered = 0;
-				if (this.commandProvider.singleClick) {
-					this.commandProvider.singleClick(node, this);
-				}
-			},
-			delay: 400
-		},
-
-			// prevent the expanding / collapsing on double click
-		beforedblclick: {
-			fn: function() {
-				return false;
-			}
-		},
-
-			// prevents label edit on a selected node
-		beforeclick: {
-			fn: function(node, event) {
-				if (!this.clicksRegistered && this.getSelectionModel().isSelected(node)) {
-					node.fireEvent('click', node, event);
-					++this.clicksRegistered;
-					return false;
-				}
-				++this.clicksRegistered;
-			}
+	constructor: function (config) {
+			// Inline label editing feature
+		this.labelEdit = config.labelEdit || this.labelEdit;
+		if (this.labelEdit ) {
+			var plugins = config.plugins || [];
+			config.plugins = plugins.concat(
+				Ext.create('TYPO3.Components.PageTree.TreeEditor', {
+					clicksToEdit: 2,
+					pluginId: 'treeEditor'
+				})
+			);
+		}
+			// Drag & drop feature
+		if (this.enableDD) {
+			config.viewConfig = Ext.applyIf(config.viewConfig || {}, this.viewConfig);
+			var plugins = config.viewConfig.plugins || [];
+			config.viewConfig.plugins = plugins.concat(
+				Ext.create('TYPO3.Components.PageTree.plugin.TreeViewDragDrop', {
+					ddGroup: config.ddGroup,
+					pluginId: 'treeViewDragDrop'
+				})
+			);
+			config.viewConfig.allowCopy = true;
 		}
+			// Call parent constructor
+		this.callParent([config]);
 	},
 
 	/**
@@ -227,28 +306,74 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	 *
 	 * @return {void}
 	 */
-	initComponent: function() {
-		if (!this.uiProvider) {
-			this.uiProvider = TYPO3.Components.PageTree.PageTreeNodeUI;
-		}
-		Ext.dd.DragDropMgr.useCache = false;
-		this.root = new Ext.tree.AsyncTreeNode(this.rootNodeConfig);
-		this.addTreeLoader();
-
-		if (this.labelEdit) {
-			this.enableInlineEditor();
-		}
-
+	initComponent: function () {
+			// Add the tree store
+		this.addStore();
+			// Add single click handler that only triggers after a delay to let the double click event
+			// a possibility to be executed (needed for label edit)
+		this.addListener('itemclick', this.onItemSingleClick, null, { delay: 400 });
+			// Init component
+		this.callParent();
+			// Drag & drop feature
 		if (this.enableDD) {
-			this.dragConfig = {ddGroup: this.ddGroup};
-			this.enableDragAndDrop();
+			this.getView().addListener('afterrender', this.enableDragAndDrop, this);
 		}
-
+			// Context menu feature
 		if (this.contextMenuProvider) {
 			this.enableContextMenu();
 		}
-
-		TYPO3.Components.PageTree.Tree.superclass.initComponent.apply(this, arguments);
+	},
+	 
+	/**
+	 * Adds the store to the tree
+	 *
+	 * @return {void}
+	 */
+	addStore: function () {
+		this.store = Ext.data.StoreManager.lookup(this.getId() + 'PageTreeStore');
+		if (!this.store) {
+			this.store = Ext.create('Ext.data.TreeStore', {
+				clearOnLoad: false,
+				listeners: {
+						// Remove nodes and add params to read operation
+					beforeload: {
+						fn: function (store, operation, options) {
+							if (operation.node) {
+								var node = operation.node;
+								node.removeAll();
+								node.commit();
+								operation.params = {
+									nodeId: node.getNodeData('id'),
+									nodeData: node.get('nodeData')
+									
+								};
+							}
+						}
+					},
+						// Restore state on initial load
+					load: {
+						fn: function (store, node, records, successful) {
+							if (successful) {
+								this.restoreState();
+							}
+						},
+						scope: this
+					}
+				},
+				model: 'TYPO3.Components.PageTree.Model',
+				nodeParam: 'nodeId',
+				proxy: {
+					type: 'direct',
+					directFn: this.treeDataProvider.getNextTreeLevel,
+					paramOrder: ['nodeId', 'nodeData'],
+					reader: {
+					    type: 'json'
+					}
+				},
+				root: this.rootNodeConfig,
+				storeId: this.getId() + 'PageTreeStore'
+			});
+		}
 	},
 
 	/**
@@ -258,9 +383,9 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	 * @param {Object} scope
 	 * return {void}
 	 */
-	refreshTree: function(callback, scope) {
-			// remove readable rootline elements while refreshing
-		if (!this.inRefreshingMode) {
+	refreshTree: function (callback, scope) {
+			// Remove readable rootline elements while refreshing
+		if (!this.store.isLoading()) {
 			var rootlineElements = Ext.select('.x-tree-node-readableRootline');
 			if (rootlineElements) {
 				rootlineElements.each(function(element) {
@@ -268,70 +393,54 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 				});
 			}
 		}
-
-		this.refreshNode(this.root, callback, scope);
+		this.refreshNode(this.getRootNode(), callback, scope);
 	},
 
 	/**
 	 * Refreshes a given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {Function} callback
 	 * @param {Object} scope
 	 * return {void}
 	 */
-	refreshNode: function(node, callback, scope) {
-		if (this.inRefreshingMode) {
-			return;
-		}
-
-		scope = scope || node;
-		this.inRefreshingMode = true;
-		var loadCallback = function(node) {
-			node.ownerTree.inRefreshingMode = false;
-			if (node.ownerTree.restoreState) {
-				node.ownerTree.restoreState(node.getPath());
-			}
-		};
-
-		if (callback) {
-			loadCallback = callback.createSequence(loadCallback);
-		}
-
-		this.getLoader().load(node, loadCallback, scope);
+	refreshNode: function (node, callback, scope) {
+		this.store.load({
+			node: node,
+			callback: callback || Ext.emptyFn,
+			scope: scope || this
+		});
 	},
 
 	/**
-	 * Adds a tree loader implementation that uses the directFn feature
+	 * Handles singe click on tree item
 	 *
-	 * return {void}
+	 * return {Boolean}
 	 */
-	addTreeLoader: function() {
-		this.loader = new Ext.tree.TreeLoader({
-			directFn: this.treeDataProvider.getNextTreeLevel,
-			paramOrder: 'nodeId,attributes',
-			nodeParameter: 'nodeId',
-			baseAttrs: {
-				uiProvider: this.uiProvider
-			},
-
-				// an id can never be zero in ExtJS, but this is needed
-				// for the root line feature or it will never be working!
-			createNode: function(attr) {
-				if (attr.id == 0) {
-					attr.id = 'siteRootNode';
-				}
-
-				return Ext.tree.TreeLoader.prototype.createNode.call(this, attr);
-			},
+	onItemSingleClick: function (view, node, item, index, event) {
+		var tree = view.panel;
+			// Check if the tree editor was triggered by dblclick
+			// If so, stop the next two clicks
+		if (tree.inhibitClicks) {
+			--tree.inhibitClicks;
+			event.stopEvent();
+			return false;
+		}
 
-			listeners: {
-				beforeload: function(treeLoader, node) {
-					treeLoader.baseParams.nodeId = node.id;
-					treeLoader.baseParams.attributes = node.attributes.nodeData;
-				}
+		if (tree.commandProvider.singleClick) {
+			tree.commandProvider.singleClick(node, tree);
+		}
+			// Fire the context menu on a single click on the node icon (Beware of drag&drop!)
+		if (!TYPO3.Components.PageTree.Configuration.disableIconLinkToContextmenu
+			|| TYPO3.Components.PageTree.Configuration.disableIconLinkToContextmenu === '0'
+		) {
+			var target = event.getTarget('span.t3-icon-apps-pagetree');
+			if (target) {
+				view.fireEvent('itemcontextmenu', view, node, item, index, event);
+				event.stopEvent();
 			}
-		});
+		}
+		return true;
 	},
 
 	/**
@@ -340,58 +449,33 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	 * return {void}
 	 */
 	enableContextMenu: function() {
-		this.contextMenu = new TYPO3.Components.PageTree.ContextMenu();
-
-		this.on('contextmenu', function(node, event) {
-			this.openContextMenu(node, event);
+		this.contextMenu = Ext.create('TYPO3.Components.PageTree.ContextMenu', { pageTree: this });
+		this.getView().on('itemcontextmenu', function (view, node, item, index, event) {
+			view.panel.openContextMenu(view, node, item, index, event);
 		});
 	},
 
 	/**
 	 * Open a context menu for the given node
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {Ext.EventObject} event
 	 * return {void}
 	 */
-	openContextMenu: function(node, event) {
-		var attributes = Ext.apply(node.attributes.nodeData, {
-			t3ContextInfo: node.ownerTree.t3ContextInfo
-		});
-
-		this.contextMenuProvider.getActionsForNodeArray(
-			attributes,
-			function(configuration) {
-				this.contextMenu.removeAll();
-				this.contextMenu.fill(node, this, configuration);
-				if (this.contextMenu.items.length) {
-					this.contextMenu.showAt(event.getXY());
-
+	openContextMenu: function(view, node, item, index, event) {
+		var tree = view.panel;
+		node.setNodeData('t3ContextInfo', tree.t3ContextInfo);
+		tree.contextMenuProvider.getActionsForNodeArray(
+			node.get('nodeData'),
+			function (configuration) {
+				tree.contextMenu.removeAll();
+				tree.contextMenu.fill(node, tree, configuration);
+				if (tree.contextMenu.items.length) {
+					tree.contextMenu.showAt(event.getXY());
 				}
-			},
-			this
+			}
 		);
-	},
-
-	/**
-	 * Initialize the inline editor for the given tree.
-	 *
-	 * @return {void}
-	 */
-	enableInlineEditor: function() {
-		this.treeEditor = new TYPO3.Components.PageTree.TreeEditor(this);
-	},
-
-	/**
-	 * Triggers the editing of the node if the tree editor is available
-	 *
-	 * @param {Ext.tree.TreeNode} node
-	 * @return {void}
-	 */
-	triggerEdit: function(node) {
-		if (this.treeEditor) {
-			this.treeEditor.triggerEdit(node);
-		}
+		event.stopEvent();
 	},
 
 	/**
@@ -400,208 +484,144 @@ TYPO3.Components.PageTree.Tree = Ext.extend(Ext.tree.TreePanel, {
 	 * return {void}
 	 */
 	enableDragAndDrop: function() {
-			// init proxy element
-		this.on('startdrag', this.initDd, this);
-		this.on('enddrag', this.stopDd, this);
-
-			// node is moved
-		this.on('movenode', this.moveNode, this);
-
-			// new node is created/copied
-		this.on('beforenodedrop', this.beforeDropNode, this);
-		this.on('nodedrop', this.dropNode, this);
-
-			// listens on the ctrl key to toggle the copy mode
-		(new Ext.KeyMap(document, {
-			key: Ext.EventObject.CONTROL,
-			scope: this,
-			buffer: 250,
-			fn: function() {
-				if (!this.controlKeyPressed && this.dragZone.dragging && this.copyHint) {
-					if (this.shouldCopyNode) {
-						this.copyHint.show();
-					} else {
-						this.copyHint.hide();
-					}
-
-					this.shouldCopyNode = !this.shouldCopyNode;
-					this.dragZone.proxy.el.toggleClass('typo3-pagetree-copy');
-				}
-				this.controlKeyPressed = true;
-			}
-		}, 'keydown'));
-
-		(new Ext.KeyMap(document, {
-			key: Ext.EventObject.CONTROL,
-			scope: this,
-			fn: function() {
-				this.controlKeyPressed = false;
-			}
-		}, 'keyup'));
-
-			// listens on the escape key to stop the dragging
-		(new Ext.KeyMap(document, {
-			key: Ext.EventObject.ESC,
-			scope: this,
-			buffer: 250,
-			fn: function(event) {
-				if (this.dragZone.dragging) {
-					Ext.dd.DragDropMgr.stopDrag(event);
-					this.dragZone.onInvalidDrop(event);
-				}
-			}
-		}, 'keydown'));
+		var view = this.getView();
+		var dragZone = view.getPlugin('treeViewDragDrop').dragZone;
+
+			// Show drop zone before drag, otherwise the proxy is never notified
+		dragZone.onBeforeDrag = Ext.Function.bind(this.startDeletionDropZone, view);
+			// Hide the drop zone after the drag completes
+		dragZone.onMouseUp = Ext.Function.bind(this.stopDeletionDropZone, view);
+		dragZone.endDrag = Ext.Function.bind(this.stopDeletionDropZone, view);
+		dragZone.afterInvalidDrop = Ext.Function.bind(this.stopDeletionDropZone, view, [true]);
+
+			// Node is moved
+		this.on('itemmove', this.moveNode, this);
+
+			// New node is created/copied
+		view.on('beforedrop', this.beforeDropNode, this);
+		view.on('drop', this.dropNode, this);
 	},
 
 	/**
-	 * Disables the deletion drop zone if configured
+	 * Enables the deletion drop zone if configured
 	 *
 	 * @return {void}
 	 */
-	stopDd: function() {
-		if (this.deletionDropZoneId) {
-			Ext.getCmp(this.deletionDropZoneId).hide();
-			this.app.doLayout();
-		}
-	},
-
-	/**
-	 * Enables the deletion drop zone if configured. Also it creates the
-	 * shown dd proxy element.
-	 *
-	 * @param {TYPO3.Components.PageTree.Tree} treePanel
-	 * @param {Ext.tree.TreeNode} node
-	 * @return {void}
-	 */
-	initDd: function(treePanel, node) {
-		var nodeHasChildNodes = (node.hasChildNodes() || node.isExpandable());
-		if (this.deletionDropZoneId &&
+	startDeletionDropZone: function (dragData, event) {
+		var view = dragData.view,
+			tree = view.panel,
+			node = view.getRecord(dragData.item),
+			nodeHasChildNodes = (node.hasChildNodes() || node.isExpandable());
+		var tree = this.panel;
+		if (tree.deletionDropZoneId &&
 			(!nodeHasChildNodes ||
 			(nodeHasChildNodes && TYPO3.Components.PageTree.Configuration.canDeleteRecursivly)
 		)) {
-			Ext.getCmp(this.deletionDropZoneId).show();
-			this.app.doLayout();
+			Ext.getCmp(tree.deletionDropZoneId).show();
 		}
-		this.initDDProxyElement();
 	},
 
 	/**
-	 * Adds the copy hint to the proxy element
+	 * Disables the deletion drop zone if configured
 	 *
 	 * @return {void}
 	 */
-	initDDProxyElement: function() {
-		this.shouldCopyNode = false;
-		this.copyHint = new Ext.Element(document.createElement('div')).addClass(this.id + '-copy');
-		this.copyHint.dom.appendChild(document.createTextNode(TYPO3.Components.PageTree.LLL.copyHint));
-		this.copyHint.setVisibilityMode(Ext.Element.DISPLAY);
-		this.dragZone.proxy.el.shadow = false;
-		this.dragZone.proxy.ghost.dom.appendChild(this.copyHint.dom);
+	stopDeletionDropZone: function (forceStop) {
+		var tree = this.panel;
+		if (tree.deletionDropZoneId && (!this.getPlugin('treeViewDragDrop').dragZone.dragging || forceStop)) {
+			Ext.getCmp(tree.deletionDropZoneId).hide();
+		}
 	},
 
 	/**
-	 * Creates a Fake Node
+	 * Creates a place holder node when a new node is about to be dropped
 	 *
-	 * This must be done to prevent the calling of the moveNode event.
-	 *
-	 * @param {object} dragElement
+	 * @param {HTMLElement node} node
+	 * @param {object} dragData
+	 * @param {TYPO3.Components.PageTree.Model} overNode
+	 * @param {string} dropPosition
+	 * @return {boolean}
 	 */
-	beforeDropNode: function(dragElement) {
-		if (dragElement.data && dragElement.data.item && dragElement.data.item.shouldCreateNewNode) {
-			this.t3ContextInfo.serverNodeType = dragElement.data.item.nodeType;
-			dragElement.dropNode = new Ext.tree.TreeNode({
+	beforeDropNode: function (node, dragData, overNode, dropPosition) {
+		if (dragData && dragData.item && dragData.item.shouldCreateNewNode) {
+				// Inserting a new node of the type that was selected in the top panel
+			this.t3ContextInfo.serverNodeType = dragData.item.nodeType;
+			dragData.dropNode = Ext.create('TYPO3.Components.PageTree.Model', {
 				text: TYPO3.Components.PageTree.LLL.fakeNodeHint,
 				leaf: true,
 				isInsertedNode: true
 			});
-
-				// fix incorrect cancel value
-			dragElement.cancel = false;
-
-		} else if (this.shouldCopyNode) {
-			dragElement.dropNode.ui.onOut();
-			var attributes = dragElement.dropNode.attributes;
-			attributes.isCopiedNode = true;
-			attributes.id = 'fakeNode';
-			dragElement.dropNode = new Ext.tree.TreeNode(attributes);
+			dragData.records = [dragData.dropNode];
 		}
-
 		return true;
 	},
 
 	/**
-	 * Differentiate between the copy and insert event
+	 * Handle the copy and insert events
 	 *
-	 * @param {Ext.tree.TreeDropZone} dragElement
+	 * @param {HTMLElement node} node
+	 * @param {object} dragData
+	 * @param {TYPO3.Components.PageTree.Model} overNode
+	 * @param {string} dropPosition
 	 * return {void}
 	 */
-	dropNode: function(dragElement) {
-		this.controlKeyPressed = false;
-		if (dragElement.dropNode.attributes.isInsertedNode) {
-			dragElement.dropNode.attributes.isInsertedNode = false;
-			this.insertNode(dragElement.dropNode);
-		} else if (dragElement.dropNode.attributes.isCopiedNode) {
-			dragElement.dropNode.attributes.isCopiedNode = false;
-			this.copyNode(dragElement.dropNode)
+	dropNode: function (node, dragData, overNode, dropPosition) {
+		if (dragData.dropNode) {
+			if (dragData.dropNode.get('isInsertedNode')) {
+				dragData.dropNode.set('isInsertedNode', false);
+				this.insertNode(dragData.dropNode);
+			}
+		} else if (dragData.copy) {
+			this.copyNode(dragData.records[0]);
 		}
 	},
 
 	/**
 	 * Moves a node
 	 *
-	 * @param {TYPO3.Components.PageTree.Tree} tree
-	 * @param {Ext.tree.TreeNode} movedNode
-	 * @param {Ext.tree.TreeNode} oldParent
-	 * @param {Ext.tree.TreeNode} newParent
+	 * @param {TYPO3.Components.PageTree.Model} movedNode
+	 * @param {TYPO3.Components.PageTree.Model} oldParent
+	 * @param {TYPO3.Components.PageTree.Model} newParent
 	 * @param {int} position
 	 * return {void}
 	 */
-	moveNode: function(tree, movedNode, oldParent, newParent, position) {
-		this.controlKeyPressed = false;
-		tree.t3ContextNode = movedNode;
-
+	moveNode: function (movedNode, oldParent, newParent, position) {
+		this.t3ContextNode = movedNode;
 		if (position === 0) {
-			this.commandProvider.moveNodeToFirstChildOfDestination(newParent, tree);
+			this.commandProvider.moveNodeToFirstChildOfDestination(newParent, this);
 		} else {
 			var previousSiblingNode = newParent.childNodes[position - 1];
-			this.commandProvider.moveNodeAfterDestination(previousSiblingNode, tree);
+			this.commandProvider.moveNodeAfterDestination(previousSiblingNode, this);
 		}
 	},
 
 	/**
 	 * Inserts a node
 	 *
-	 * @param {Ext.tree.TreeNode} movedNode
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * return {void}
 	 */
-	insertNode: function(movedNode) {
-		this.t3ContextNode = movedNode.parentNode;
-
-		movedNode.disable();
-		if (movedNode.previousSibling) {
-			this.commandProvider.insertNodeAfterDestination(movedNode, this);
+	insertNode: function (node) {
+		this.t3ContextNode = node.parentNode;
+		if (node.previousSibling) {
+			this.commandProvider.insertNodeAfterDestination(node, this);
 		} else {
-			this.commandProvider.insertNodeToFirstChildOfDestination(movedNode, this);
+			this.commandProvider.insertNodeToFirstChildOfDestination(node, this);
 		}
 	},
 
 	/**
 	 * Copies a node
 	 *
-	 * @param {Ext.tree.TreeNode} movedNode
+	 * @param {TYPO3.Components.PageTree.Model} movedNode
 	 * return {void}
 	 */
-	copyNode: function(movedNode) {
-		this.t3ContextNode = movedNode;
-
-		movedNode.disable();
-		if (movedNode.previousSibling) {
-			this.commandProvider.copyNodeAfterDestination(movedNode, this);
+	copyNode: function (node) {
+		this.t3ContextNode = node;
+		if (node.previousSibling) {
+			this.commandProvider.copyNodeAfterDestination(node, this);
 		} else {
-			this.commandProvider.copyNodeToFirstChildOfDestination(movedNode, this);
+			this.commandProvider.copyNodeToFirstChildOfDestination(node, this);
 		}
 	}
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.Tree', TYPO3.Components.PageTree.Tree);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/treedropzone.js b/t3lib/js/extjs/components/pagetree/javascript/treedropzone.js
new file mode 100644
index 0000000000000000000000000000000000000000..2f4888ca9ffc77976915ab8180120f1bd2ac925f
--- /dev/null
+++ b/t3lib/js/extjs/components/pagetree/javascript/treedropzone.js
@@ -0,0 +1,112 @@
+/***************************************************************
+*  Copyright notice
+*
+*  (c) 2011 Stanislas Rolland <typo3@sjbr.ca>
+*  All rights reserved
+*
+*  This script is part of the TYPO3 project. The TYPO3 project is
+*  free software; you can redistribute it and/or modify
+*  it under the terms of the GNU General Public License as published by
+*  the Free Software Foundation; either version 2 of the License, or
+*  (at your option) any later version.
+*
+*  The GNU General Public License can be found at
+*  http://www.gnu.org/copyleft/gpl.html.
+*  A copy is found in the textfile GPL.txt and important notices to the license
+*  from the author is found in LICENSE.txt distributed with these scripts.
+*
+*
+*  This script is distributed in the hope that it will be useful,
+*  but WITHOUT ANY WARRANTY; without even the implied warranty of
+*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+*  GNU General Public License for more details.
+*
+*  This copyright notice MUST APPEAR in all copies of the script!
+***************************************************************/
+/**
+ * @class TYPO3.Components.PageTree.ViewDropZone
+ *
+ * handleNodeDrop method is modified in order to process copied nodes
+ * 
+ * Based on ExtJS 4.0.7.
+ * Should be reviewed on ExtJS upgrade
+ *
+ * @namespace TYPO3.Components.PageTree
+ * @extends Ext.tree.ViewDropZone
+ * @author Stanislas Rolland <typo3@sjbr.ca>
+ */
+Ext.define('TYPO3.Components.PageTree.ViewDropZone', {
+	extend: 'Ext.tree.ViewDropZone',
+
+	handleNodeDrop: function(data, targetNode, position) {
+		var me = this,
+			view = me.view,
+			parentNode = targetNode.parentNode,
+			store = view.getStore(),
+			recordDomNodes = [],
+			records, i, len,
+			insertionMethod, argList,
+			needTargetExpand,
+			transferData,
+			processDrop;
+
+		if (data.copy) {
+			records = data.records;
+			data.records = [];
+			for (i = 0, len = records.length; i < len; i++) {
+				data.records.push(Ext.apply({}, records[i].data));
+			}
+		}
+
+		me.cancelExpand();
+
+		if (position == 'before') {
+			insertionMethod = parentNode.insertBefore;
+			argList = [null, targetNode];
+			targetNode = parentNode;
+		} else if (position == 'after') {
+			if (targetNode.nextSibling) {
+				insertionMethod = parentNode.insertBefore;
+				argList = [null, targetNode.nextSibling];
+			} else {
+				insertionMethod = parentNode.appendChild;
+				argList = [null];
+			}
+			targetNode = parentNode;
+		} else {
+			if (!targetNode.isExpanded()) {
+				needTargetExpand = true;
+			}
+			insertionMethod = targetNode.appendChild;
+			argList = [null];
+		}
+
+		transferData = function() {
+			var node;
+			for (i = 0, len = data.records.length; i < len; i++) {
+				argList[0] = data.records[i];
+				node = insertionMethod.apply(targetNode, argList);
+					// We need to update the records array in order to process the copied nodes in the drop event handler
+				data.records[i] = node;
+
+				if (Ext.enableFx && me.dropHighlight) {
+					recordDomNodes.push(view.getNode(node));
+				}
+			}
+
+			if (Ext.enableFx && me.dropHighlight) {
+				Ext.Array.forEach(recordDomNodes, function(n) {
+					if (n) {
+						Ext.fly(n.firstChild ? n.firstChild : n).highlight(me.dropHighlightColor);
+					}
+				});
+			}
+		};
+
+		if (needTargetExpand) {
+			targetNode.expand(false, transferData);
+		} else {
+			transferData();
+		}
+	}
+});
\ No newline at end of file
diff --git a/t3lib/js/extjs/components/pagetree/javascript/treeeditor.js b/t3lib/js/extjs/components/pagetree/javascript/treeeditor.js
index 0d12e7d526678b227e3039e4a31f56e3f38e3910..5daa34e2d79ec015ad0316cc427fe8cce03d1b52 100644
--- a/t3lib/js/extjs/components/pagetree/javascript/treeeditor.js
+++ b/t3lib/js/extjs/components/pagetree/javascript/treeeditor.js
@@ -23,8 +23,6 @@
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/
-Ext.namespace('TYPO3.Components.PageTree');
-
 /**
  * @class TYPO3.Components.PageTree.TreeEditor
  *
@@ -32,30 +30,46 @@ Ext.namespace('TYPO3.Components.PageTree');
  * editable label.
  *
  * @namespace TYPO3.Components.PageTree
- * @extends Ext.tree.TreeEditor
+ * @extends Ext.ux.tree.TreeEditing
  * @author Stefan Galinski <stefan.galinski@gmail.com>
  */
-TYPO3.Components.PageTree.TreeEditor = Ext.extend(Ext.tree.TreeEditor, {
-	/**
-	 * Don't send any save events if the value wasn't changed
-	 *
-	 * @type {Boolean}
-	 */
-	ignoreNoChange: false,
+Ext.define('TYPO3.Components.PageTree.TreeEditor', {
+	extend: 'Ext.grid.plugin.CellEditing',
+	alias: 'plugin.pagetreeeditor',
 
 	/**
-	 * Edit delay
-	 *
-	 * @type {int}
+	 * @override
+	 * @private Collects all information necessary for any subclasses to perform their editing functions.
+	 * @param record
+	 * @param columnHeader
+	 * @returns {Object} The editing context based upon the passed record and column
 	 */
-	editDelay: 250,
+	getEditingContext: function (record, columnHeader) {
+	 	var me = this,
+	 		grid = me.grid,
+	 		store = grid.store,
+	 		colIdx,
+	 		editor,
+	 		originalValue,
+	 		value;
 
-	/**
-	 * Indicates if an underlying shadow should be shown
-	 *
-	 * @type {Boolean}
-	 */
-	shadow: false,
+		if (Ext.isNumber(columnHeader)) {
+			colIdx = columnHeader;
+			columnHeader = grid.headerCt.getHeaderAtIndex(colIdx);
+		} else {
+			colIdx = columnHeader.getIndex();
+		}
+
+		return {
+			column: columnHeader,
+			colIdx: colIdx,
+			field: columnHeader.dataIndex,
+			grid: grid,
+			originalValue: record.getNodeData('editableText'),
+			record: record,
+			tree: grid
+		};
+	},
 
 	/**
 	 * Listeners
@@ -63,80 +77,66 @@ TYPO3.Components.PageTree.TreeEditor = Ext.extend(Ext.tree.TreeEditor, {
 	 * Handles the synchronization between the edited label and the shown label.
 	 */
 	listeners: {
-		beforecomplete: function(node) {
-			this.updatedValue = this.getValue();
-			if (this.updatedValue === '') {
-				this.cancelEdit();
-				return false;
-			}
-			this.setValue(this.editNode.attributes.prefix + Ext.util.Format.htmlEncode(this.updatedValue) + this.editNode.attributes.suffix);
-		},
-
-		complete: {
-			fn: function(node, newValue, oldValue) {
-				if (newValue === oldValue) {
-					this.fireEvent('canceledit', this);
+		beforeedit: {
+			fn: function (editEvent) {
+				var tree = editEvent.tree;
+					// Prevent editing the currently selected node
+					// Prevent editing if the node is not editable
+				if (editEvent.record == tree.currentSelectedNode || !editEvent.record.getNodeData('editable')) {
+					if (tree.currentSelectedNode) {
+						tree.getView().select(tree.currentSelectedNode);
+					}
 					return false;
 				}
-
-				this.editNode.getOwnerTree().commandProvider.saveTitle(node, this.updatedValue, oldValue, this);
+					// Inhibit clicks on the tree while editing
+				tree.inhibitClicks = 2;
 			}
 		},
-
-		startEdit: {
-			fn: function(element, value) {
-				this.field.selectText();
+		validateedit: {
+			fn: function (treeEditor, editEvent) {
+				var editorField = treeEditor.getEditor(editEvent.record, editEvent.column);
+				this.newValue = editorField.getValue();
+				if (this.newValue === '' || this.newValue === editEvent.originalValue) {
+					var tree = editEvent.tree;
+					if (tree.currentSelectedNode) {
+						tree.getView().select(tree.currentSelectedNode);
+					}
+					return false;
+				} else {
+					editorField.setValue(editEvent.record.getNodeData('prefix') + Ext.util.Format.htmlEncode(this.newValue) + editEvent.record.getNodeData('suffix'));
+				}
+				
 			}
 		},
-
-		canceledit: function() {
-			var tree = this.editNode.getOwnerTree();
-			if (tree.currentSelectedNode) {
-				tree.currentSelectedNode.select();
+		edit: {
+			fn: function (treeEditor, editEvent) {
+				var tree = editEvent.tree;
+				tree.commandProvider.saveTitle(editEvent.record, this.newValue, editEvent.originalValue, treeEditor, tree, editEvent.field);
 			}
 		}
 	},
+	cancelEdit: function () {
+		var tree = this.grid;
+		if (tree.currentSelectedNode) {
+			tree.getView().select(tree.currentSelectedNode);
+		}
+		this.callParent(arguments);
+	},
 
 	/**
-	 * Updates the edit node
+	 * Updates the text field
 	 *
-	 * @param {Ext.tree.TreeNode} node
+	 * @param {TYPO3.Components.PageTree.Model} node
 	 * @param {String} editableText
-	 * @param {String} updatedNode
+	 * @param {String} updatedText
+	 * @param {String} dataIndex
+	 * @param {TYPO3.Components.PageTree.Tree} tree
 	 * @return {void}
 	 */
-	updateNodeText: function(node, editableText, updatedNode) {
-		this.editNode.setText(this.editNode.attributes.prefix + updatedNode + this.editNode.attributes.suffix);
-		this.editNode.attributes.editableText = editableText;
-	},
-
-	/**
-	 * Overridden method to set another editable text than the node text attribute
-	 *
-	 * @param {Ext.tree.TreeNode} node
-	 * @return {Boolean}
-	 */
-	triggerEdit: function(node) {
-		this.completeEdit();
-		if (node.attributes.editable !== false) {
-			this.editNode = node;
-			if (this.tree.autoScroll) {
-				Ext.fly(node.ui.getEl()).scrollIntoView(this.tree.body);
-			}
-
-			var value = node.text || '';
-			if (!Ext.isGecko && Ext.isEmpty(node.text)) {
-				node.setText(' ');
-			}
-
-				// TYPO3 MODIFICATION to use another attribute
-			value = node.attributes.editableText;
-
-			this.autoEditTimer = this.startEdit.defer(this.editDelay, this, [node.ui.textNode, value]);
-			return false;
-		}
+	updateNodeText: function (record, editableText, updatedText, dataIndex, tree) {
+		record.set(dataIndex, record.getNodeData('prefix') + updatedText + record.getNodeData('suffix'));
+		record.setNodeData('editableText', editableText);
+		record.commit();
+		tree.getView().refresh(record.getId());
 	}
 });
-
-// XTYPE Registration
-Ext.reg('TYPO3.Components.PageTree.TreeEditor', TYPO3.Components.PageTree.TreeEditor);
diff --git a/t3lib/js/extjs/components/pagetree/javascript/treeviewdragdrop.js b/t3lib/js/extjs/components/pagetree/javascript/treeviewdragdrop.js
new file mode 100644
index 0000000000000000000000000000000000000000..b0c5bee20c5fd2d65579c48756e0d8d9ca9e1987
--- /dev/null
+++ b/t3lib/js/extjs/components/pagetree/javascript/treeviewdragdrop.js
@@ -0,0 +1,65 @@
+/***************************************************************
+*  Copyright notice
+*
+*  (c) 2011 Stanislas Rolland <typo3@sjbr.ca>
+*  All rights reserved
+*
+*  This script is part of the TYPO3 project. The TYPO3 project is
+*  free software; you can redistribute it and/or modify
+*  it under the terms of the GNU General Public License as published by
+*  the Free Software Foundation; either version 2 of the License, or
+*  (at your option) any later version.
+*
+*  The GNU General Public License can be found at
+*  http://www.gnu.org/copyleft/gpl.html.
+*  A copy is found in the textfile GPL.txt and important notices to the license
+*  from the author is found in LICENSE.txt distributed with these scripts.
+*
+*
+*  This script is distributed in the hope that it will be useful,
+*  but WITHOUT ANY WARRANTY; without even the implied warranty of
+*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+*  GNU General Public License for more details.
+*
+*  This copyright notice MUST APPEAR in all copies of the script!
+***************************************************************/
+/**
+ * @class TYPO3.Components.PageTree.plugin.TreeViewDragDrop
+ *
+ * onViewRender method is modified in order to use TYPO3.Components.PageTree.ViewDropZone
+ *
+ * @namespace TYPO3.Components.PageTree
+ * @extends Ext.tree.plugin.TreeViewDragDrop
+ * @author Stanislas Rolland <typo3@sjbr.ca>
+ */
+Ext.define('TYPO3.Components.PageTree.plugin.TreeViewDragDrop', {
+    extend: 'Ext.tree.plugin.TreeViewDragDrop',
+    alias: 'plugin.pagetreeviewdragdrop',
+
+    onViewRender: function(view) {
+        var me = this;
+
+        if (me.enableDrag) {
+            me.dragZone = Ext.create('Ext.tree.ViewDragZone', {
+                view: view,
+                ddGroup: me.dragGroup || me.ddGroup,
+                dragText: me.dragText,
+                repairHighlightColor: me.nodeHighlightColor,
+                repairHighlight: me.nodeHighlightOnRepair
+            });
+        }
+
+        if (me.enableDrop) {
+            me.dropZone = Ext.create('TYPO3.Components.PageTree.ViewDropZone', {
+                view: view,
+                ddGroup: me.dropGroup || me.ddGroup,
+                allowContainerDrops: me.allowContainerDrops,
+                appendOnly: me.appendOnly,
+                allowParentInserts: me.allowParentInserts,
+                expandDelay: me.expandDelay,
+                dropHighlightColor: me.nodeHighlightColor,
+                dropHighlight: me.nodeHighlightOnDrop
+            });
+        }
+    }
+});
\ No newline at end of file
diff --git a/t3lib/js/extjs/tree/tree.js b/t3lib/js/extjs/tree/tree.js
index f0cad3962071883b4f1ad80753c0d669723cca24..bbd262485b33cf8b8a1eb74c06bb3f87a23ecf1e 100644
--- a/t3lib/js/extjs/tree/tree.js
+++ b/t3lib/js/extjs/tree/tree.js
@@ -23,60 +23,64 @@
  *
  *  This copyright notice MUST APPEAR in all copies of the script!
  ***************************************************************/
-
-Ext.ns('TYPO3.Components', 'TYPO3.Components.Tree');
-
 /**
  * TYPO3window - General TYPO3 tree component
  */
+Ext.ns('TYPO3.Components.Tree');
 
 TYPO3.Components.Tree = {};
 TYPO3.Components.Tree.StandardTreeItemData = [];
 
-TYPO3.Components.Tree.StandardTree = function(config) {
-	var conf = Ext.apply({
-		header: false,
-		width: 280,
-		rootVisible: false,
-		useArrows: false,
-		lines: true,
-		autoScroll: true,
-		containerScroll: true,
-		exclusiveSelectedKey: null,
-		stateful: true,
-		filterOptionStartsWith: true,
-		countSelectedNodes: 0,
-		loader: new Ext.tree.TreeLoader({
-			preloadChildren: true,
-			clearOnLoad: false
-		}),
-		root: new Ext.tree.AsyncTreeNode({
-			text: TYPO3.l10n.localize('tcatree'),
-			id: 'root',
-			expanded: true,
-			children: TYPO3.Components.Tree.StandardTreeItemData[config.id]
-		}),
-		collapseFirst: false,
-		listeners: {
-			'checkchange': function(checkedNode, checked) {
-				if (Ext.isFunction(this.checkChangeHandler)) {
-					this.checkChangeHandler.call(this, checkedNode, checked);
-				}
-			},
-			scope: this
-		}
-	}, config);
-	TYPO3.Components.Tree.StandardTree.superclass.constructor.call(this, conf);
-};
-
-
-Ext.extend(TYPO3.Components.Tree.StandardTree, Ext.tree.TreePanel, {
-
+Ext.define('TYPO3.Components.Tree.StandardTree', {
+	extend: 'Ext.tree.Panel',
+	
+	constructor: function(config) {
+		var conf = Ext.apply({
+			header: false,
+			width: 280,
+			rootVisible: false,
+			useArrows: false,
+			lines: true,
+			autoScroll: true,
+			containerScroll: true,
+			exclusiveSelectedKey: null,
+			stateful: true,
+			filterOptionStartsWith: true,
+			countSelectedNodes: 0,
+				// Needs to be migrated to ExtJS 4
+			loader: new Ext.tree.TreeLoader({
+				preloadChildren: true,
+				clearOnLoad: false
+			}),
+			root: new Ext.tree.AsyncTreeNode({
+				text: TYPO3.l10n.localize('tcatree'),
+				id: 'root',
+				expanded: true,
+				children: TYPO3.Components.Tree.StandardTreeItemData[config.id]
+			}),
+			collapseFirst: false,
+			listeners: {
+				'checkchange': function(checkedNode, checked) {
+					if (Ext.isFunction(this.checkChangeHandler)) {
+						this.checkChangeHandler.call(this, checkedNode, checked);
+					}
+				},
+				scope: this
+			}
+		}, config);
+		this.callParent([conf]);
+	},
 	initComponent: function() {
-		Ext.apply(this, {
-			tbar: this.initialConfig.showHeader ? TYPO3.Components.Tree.Toolbar([], this) : null
-		});
-		TYPO3.Components.Tree.StandardTree.superclass.initComponent.call(this);
+		if (this.initialConfig.showHeader) {
+			Ext.apply(this, {
+				dockedItems: [{
+					xtype: 'toolbar',
+					dock: 'top',
+					items: TYPO3.Components.Tree.Toolbar([], this)
+				}]
+			});
+		}
+		this.callParent(arguments);
 	},
 	filterTree: function(filterText) {
 		var text = filterText.getValue();
@@ -139,7 +143,7 @@ TYPO3.Components.Tree.Toolbar = function(items, scope) {
 				]
 			}
 		},
-		new Ext.form.TextField({
+		Ext.create('Ext.form.field.Text', {
 			width: 150,
 			emptyText: TYPO3.l10n.localize('tcatree.findItem'),
 			enableKeyEvents: true,
diff --git a/t3lib/tree/class.t3lib_tree_abstracttree.php b/t3lib/tree/class.t3lib_tree_abstracttree.php
index cda092584e1993f431ccdf0eb20bccfde018190e..5ba1ae315e535bf86c182c76f46b4cc768644d70 100644
--- a/t3lib/tree/class.t3lib_tree_abstracttree.php
+++ b/t3lib/tree/class.t3lib_tree_abstracttree.php
@@ -63,7 +63,7 @@ abstract class t3lib_tree_AbstractTree {
 	}
 
 	/**
-	 * @param t3lib_tree_renderer_Abstract $dataProvider
+	 * @param t3lib_tree_renderer_Abstract $nodeRenderer
 	 * @return void
 	 */
 	public function setNodeRenderer(t3lib_tree_renderer_Abstract $nodeRenderer) {
@@ -83,6 +83,15 @@ abstract class t3lib_tree_AbstractTree {
 	 * @return t3lib_tree_Node
 	 */
 	abstract public function getRoot();
+
+	/**
+	 * @param mixed $search
+	 * @return t3lib_tree_Node
+	 */
+	public function find($search) {
+		return $this->getRoot()->find($search);
+	}
+
 }
 
 ?>
\ No newline at end of file
diff --git a/t3lib/tree/class.t3lib_tree_node.php b/t3lib/tree/class.t3lib_tree_node.php
index dd098882ecb3121dd4783326f2abe88349bc45d7..319307ccba22baec40ff4c221d11cc392f07e206 100644
--- a/t3lib/tree/class.t3lib_tree_node.php
+++ b/t3lib/tree/class.t3lib_tree_node.php
@@ -62,7 +62,6 @@ class t3lib_tree_Node implements t3lib_tree_ComparableNode, Serializable {
 	 * This is useful for the deserialization.
 	 *
 	 * @param array $data
-	 * @return void
 	 */
 	public function __construct(array $data = array()) {
 		if (count($data)) {
@@ -107,7 +106,7 @@ class t3lib_tree_Node implements t3lib_tree_ComparableNode, Serializable {
 	 * @return boolean
 	 */
 	public function hasChildNodes() {
-		if ($this->childNodes !== NULL) {
+		if ($this->childNodes != NULL && $this->childNodes->count() > 0) {
 			return TRUE;
 		}
 
@@ -214,7 +213,7 @@ class t3lib_tree_Node implements t3lib_tree_ComparableNode, Serializable {
 	 * @param array $data
 	 * @return void
 	 */
-	public function dataFromArray($data) {
+	public function dataFromArray(array $data) {
 		$this->setId($data['id']);
 
 		if (isset($data['parentNode']) && $data['parentNode'] !== '') {
@@ -255,6 +254,43 @@ class t3lib_tree_Node implements t3lib_tree_ComparableNode, Serializable {
 		}
 		$this->dataFromArray($arrayRepresentation);
 	}
+
+	/**
+	 * searches the tree to find an node
+	 *
+	 * @param mixed $search
+	 * @return t3lib_tree_Node
+	 */
+	public function find($search) {
+		$result = NULL;
+
+		if($this->getId() == $search) {
+			$result = $this;
+		} elseif($this->hasChildNodes() && $this->getChildNodes() != NULL) {
+			foreach ($this->getChildNodes() AS $childNode) {
+				$result = $childNode->find($search);
+				if ($result != NULL) {
+					break;
+				}
+			}
+		}
+
+		return $result;
+	}
+
+	public function getLevel() {
+		if ($this->getParentNode() == NULL) {
+			return 0;
+		} else {
+			$i = 1;
+			$node = $this->getParentNode();
+			while ($node != NULL) {
+				$node = $node->getParentNode();
+				$i++;
+			}
+			return $i;
+		}
+	}
 }
 
 ?>
\ No newline at end of file
diff --git a/t3lib/tree/class.t3lib_tree_nodecollection.php b/t3lib/tree/class.t3lib_tree_nodecollection.php
index 266f11ee98d30b19ac98933992077fe686390f04..e1a9293a452fa15d733f6777ccdfe3f57d0e7013 100644
--- a/t3lib/tree/class.t3lib_tree_nodecollection.php
+++ b/t3lib/tree/class.t3lib_tree_nodecollection.php
@@ -62,7 +62,9 @@ class t3lib_tree_NodeCollection extends ArrayObject {
 	 *
 	 * @noapi
 	 * @see t3lib_tree_Node::compareTo
-	 * @return void
+	 * @param t3lib_tree_Node $node
+	 * @param t3lib_tree_Node $otherNode
+	 * @return int
 	 */
 	public function nodeCompare(t3lib_tree_Node $node, t3lib_tree_Node $otherNode) {
 		return $node->compareTo($otherNode);
diff --git a/t3lib/tree/class.t3lib_tree_representationnode.php b/t3lib/tree/class.t3lib_tree_representationnode.php
index 8a9482a393547d87ed54adfa248c535a64c815dd..4c64a733ad1ead5fd1439334e51aa08ba6c604ef 100644
--- a/t3lib/tree/class.t3lib_tree_representationnode.php
+++ b/t3lib/tree/class.t3lib_tree_representationnode.php
@@ -32,6 +32,7 @@
  * @author Steffen Ritter <info@steffen-ritter.net>
  * @package TYPO3
  * @subpackage t3lib
+ * @deprecated Not needed anymore as of 4.7, will be removed on 4.9
  */
 class t3lib_tree_RepresentationNode extends t3lib_tree_Node {
 	/**
@@ -69,6 +70,10 @@ class t3lib_tree_RepresentationNode extends t3lib_tree_Node {
 	 */
 	protected $callbackAction = '';
 
+	public function __construct() {
+		t3lib_div::deprecationLog("The Class t3lib_tree_RepresentationNode is deprecated as of TYPO3 4.7, will be removed in 4.9");
+	}
+
 	/**
 	 * @param string $class
 	 * @return void
diff --git a/t3lib/tree/extjs/class.t3lib_tree_extjs_abstractextjstree.php b/t3lib/tree/extjs/class.t3lib_tree_extjs_abstractextjstree.php
new file mode 100644
index 0000000000000000000000000000000000000000..b7762907bc2372b586ac38a30c6ad60ffdb5e6a0
--- /dev/null
+++ b/t3lib/tree/extjs/class.t3lib_tree_extjs_abstractextjstree.php
@@ -0,0 +1,48 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Abstract ExtJS tree based on ExtDirect
+ *
+ * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+abstract class t3lib_tree_ExtJs_AbstractExtJsTree extends t3lib_tree_AbstractTree {
+
+	/**
+	 * Fetches the next tree level
+	 *
+	 * @abstract
+	 * @param int $nodeId
+	 * @param stdClass $nodeData
+	 * @return array
+	 */
+	abstract public function getNextTreeLevel($nodeId, $nodeData);
+}
+
+?>
\ No newline at end of file
diff --git a/t3lib/tree/extdirect/class.t3lib_tree_extdirect_abstractextjstree.php b/t3lib/tree/extjs/class.t3lib_tree_extjs_abstractstatefulextjstree.php
similarity index 83%
rename from t3lib/tree/extdirect/class.t3lib_tree_extdirect_abstractextjstree.php
rename to t3lib/tree/extjs/class.t3lib_tree_extjs_abstractstatefulextjstree.php
index 8dcfe320b41001893cf2089b6cbc5e67657f49e3..6ac45d21d8bb46daad4847abc2fcf459d3c22e9f 100644
--- a/t3lib/tree/extdirect/class.t3lib_tree_extdirect_abstractextjstree.php
+++ b/t3lib/tree/extjs/class.t3lib_tree_extjs_abstractstatefulextjstree.php
@@ -27,12 +27,13 @@
 
 /**
  * Abstract ExtJS tree based on ExtDirect
+ * extends with StateProvider
  *
- * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @author Steffen Ritter
  * @package TYPO3
  * @subpackage t3lib
  */
-abstract class t3lib_tree_ExtDirect_AbstractExtJsTree extends t3lib_tree_AbstractTree {
+abstract class t3lib_tree_ExtJs_AbstractStatefulExtJsTree extends t3lib_tree_ExtJs_AbstractExtJsTree {
 
 	/**
 	 * State Provider
@@ -56,15 +57,6 @@ abstract class t3lib_tree_ExtDirect_AbstractExtJsTree extends t3lib_tree_Abstrac
 		return $this->stateProvider;
 	}
 
-	/**
-	 * Fetches the next tree level
-	 *
-	 * @abstract
-	 * @param int $nodeId
-	 * @param stdClass $nodeData
-	 * @return array
-	 */
-	abstract public function getNextTreeLevel($nodeId, $nodeData);
 }
 
 ?>
\ No newline at end of file
diff --git a/t3lib/tree/extjs/class.t3lib_tree_extjs_node.php b/t3lib/tree/extjs/class.t3lib_tree_extjs_node.php
new file mode 100644
index 0000000000000000000000000000000000000000..517f306611cb2164196bdfae0f07b59e0c608155
--- /dev/null
+++ b/t3lib/tree/extjs/class.t3lib_tree_extjs_node.php
@@ -0,0 +1,590 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Node for the usage with ExtDirect and ExtJS
+ *
+ * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+class t3lib_tree_ExtJs_Node extends t3lib_tree_Node implements t3lib_tree_RenderableNode {
+
+	/**
+	 * Indicator if the node is allowDrag
+	 *
+	 * @var bool
+	 */
+	protected $allowDrag = TRUE;
+
+	/**
+	 * Indicator if the node is allowed as a drop target
+	 *
+	 * @var bool
+	 */
+	protected $allowDrop = TRUE;
+
+	/**
+	 * Indicator if the noded is checked
+	 * three-state: TRUE|FALSE|NULL
+	 * NULL won't show a checkbox
+	 *
+	 * @var boolean|NULL
+	 */
+	protected $checked = NULL;
+
+	/**
+	 * CSS Class
+	 *
+	 * @var string
+	 */
+	protected $cls = '';
+
+	/**
+	 * The number of parents this node has. ...
+	 *
+	 * @var int
+	 */
+	protected $depth;
+
+	/**
+	 * Indicator if the node can be expanded
+	 *
+	 * @var bool
+	 */
+	protected $expandable = TRUE;
+
+	/**
+	 * Indicator if the node is expanded
+	 *
+	 * @var bool
+	 */
+	protected $expanded = FALSE;
+
+	/**
+	 * An URL for a link that's created when this config is specified.
+	 *
+	 * @var string|NULL
+	 */
+	protected $href = NULL;
+
+	/**
+	 * Target for link in $href
+	 *
+	 * @var string
+	 */
+	protected $hrefTarget = '';
+
+	/**
+	 * URL for this node's icon file
+	 *
+	 * @var string
+	 */
+	protected $icon;
+
+	/**
+	 * URL for this node's icon css class
+	 *
+	 * @var string
+	 */
+	protected $iconCls;
+
+
+	/**
+	 * The position of the node inside its parents
+	 * starting from 0
+	 *
+	 * @var int
+	 */
+	protected $index;
+
+
+	/**
+	 * True if this is the first node.
+	 *
+	 * @var boolean
+	 */
+	protected $isFirst;
+
+	/**
+	 * True if this is the last node
+	 */
+	protected $isLast;
+
+	/**
+	 * Leaf Node Indicator
+	 *
+	 * @var bool
+	 */
+	protected $leaf = TRUE;
+
+	/**
+	 * Quick Tip
+	 *
+	 * @var string
+	 */
+	protected $qtip = '';
+
+	/**
+	 * Quick Tip title
+	 *
+	 * @var string
+	 */
+	protected $qtitle = '';
+
+	/**
+	 * is root node?
+	 *
+	 * @var boolean
+	 */
+	protected $root = FALSE;
+
+	/**
+	 * Label
+	 *
+	 * @var string
+	 */
+	protected $text = '';
+
+	/**
+	 * Sets the leaf node indicator
+	 *
+	 * @param bool $isLeaf
+	 * @return void
+	 */
+	public function setLeaf($isLeaf) {
+		$this->leaf = ($isLeaf == TRUE);
+	}
+
+	/**
+	 * Returns if the node is a leaf node
+	 *
+	 * @return bool
+	 */
+	public function isLeafNode() {
+		return $this->leaf;
+	}
+
+	/**
+	 * Sets the expandable indicator
+	 *
+	 * @param bool $expandable
+	 * @return void
+	 */
+	public function setExpandable($expandable) {
+		$this->expandable = ($expandable == TRUE);
+	}
+
+	/**
+	 * Returns the expandable indicator
+	 *
+	 * @return bool
+	 */
+	public function isExpandable() {
+		return $this->expandable;
+	}
+
+	/**
+	 * Sets the expanded indicator
+	 *
+	 * @param bool $expanded
+	 * @return void
+	 */
+	public function setExpanded($expanded) {
+		$this->expanded = ($expanded == TRUE);
+	}
+
+	/**
+	 * Returns the expanded indicator
+	 *
+	 * @return bool
+	 */
+	public function isExpanded() {
+		if ($this->isLeafNode()) {
+			return TRUE;
+		}
+
+		return $this->expanded;
+	}
+
+	/**
+	 * Sets the allowDrag indicator
+	 *
+	 * @param bool $draggable
+	 * @return void
+	 */
+	public function setAllowDrag($draggable) {
+		$this->allowDrag = ($draggable == TRUE);
+	}
+
+	/**
+	 * Returns the allowDrag indicator
+	 *
+	 * @return bool
+	 */
+	public function isDraggable() {
+		return $this->allowDrag;
+	}
+
+	/**
+	 * Sets the indicator if the node can be a drop target
+	 *
+	 * @param bool $isDropTarget
+	 * @return void
+	 */
+	public function setAllowDrop($isDropTarget) {
+		$this->allowDrop = ($isDropTarget == TRUE);
+	}
+
+	/**
+	 * Returns the indicator if the node is a drop target
+	 *
+	 * @return bool
+	 */
+	public function allowDrop() {
+		return $this->allowDrop;
+	}
+
+	/**
+	 * Sets the label of the node with the source field and the prefix
+	 *
+	 * @param string $text
+	 * @return void
+	 */
+	public function setText($text) {
+		$this->text = $text;
+	}
+
+	/**
+	 * Returns the label
+	 *
+	 * @return string
+	 */
+	public function getText() {
+		return $this->text;
+	}
+
+	/**
+	 * Sets the css class(es)
+	 *
+	 * @param string $class
+	 * @return void
+	 */
+	public function setCls($class) {
+		$this->cls = $class;
+	}
+
+	/**
+	 * Returns the css class(es)
+	 *
+	 * @return string
+	 */
+	public function getCls() {
+		return $this->cls;
+	}
+
+	/**
+	 * Sets the quick tip
+	 *
+	 * @param string $qtip
+	 * @return void
+	 */
+	public function setQTip($qtip) {
+		$this->qtip = $qtip;
+	}
+
+	/**
+	 * Returns the quick tip
+	 *
+	 * @return string
+	 */
+	public function getQTip() {
+		return $this->qtip;
+	}
+
+	/**
+	 * Sets the child nodes collection
+	 *
+	 * @param t3lib_tree_NodeCollection $childNodes
+	 * @return void
+	 */
+	public function setChildNodes(t3lib_tree_NodeCollection $childNodes) {
+		parent::setChildNodes($childNodes);
+
+		if ($childNodes->count()) {
+			$this->setLeaf(FALSE);
+		}
+	}
+
+	/**
+	 * Returns the node in an array representation that can be used for serialization
+	 *
+	 * @param bool $addChildNodes
+	 * @return array
+	 */
+	public function toArray($addChildNodes = TRUE) {
+		$arrayRepresentation = array(
+			'serializeClassName' => get_class($this),
+			'allowDrag' => $this->isDraggable(),
+			'allowDrop' => $this->allowDrop(),
+			'checked'	=> $this->getChecked(),
+			'cls' => $this->getCls(),
+			'depth' => $this->getDepth(),
+			'expandable' => $this->isExpandable(),
+			'expanded' => $this->isExpanded(),
+			'href' => $this->getHref(),
+			'hrefTarget' => $this->getHrefTarget(),
+			'id' => $this->getId(),
+			'icon' => $this->getIcon(),
+			'iconCls' => $this->getIconCls(),
+			'index' => $this->getIndex(),
+			'isFirst' => $this->getIsFirst(),
+			'isLast' => $this->getIsLast(),
+			'parentId' => (!$this->getRoot() && $this->getParentNode() ? $this->getParentNode()->getId() : ''),
+			'qtip' => $this->getQTip(),
+			'qtitle' => $this->getQtitle(),
+			'root' => $this->getRoot(),
+			'text' => $this->getText(),
+		);
+
+			// do not provide config if empty
+		if (trim($this->href) == '') {
+			unset($arrayRepresentation['href']);
+			unset($arrayRepresentation['hrefTarget']);
+		}
+
+			// only added check option if checkbox should be shown
+		if($this->getChecked() == NULL) {
+			unset($arrayRepresentation['checked']);
+		}
+			// only set the leaf attribute if the node has children's,
+			// otherwise you cannot add child's to real leaf nodes
+		if (!$this->isLeafNode()) {
+			$arrayRepresentation['leaf'] = FALSE;
+		}
+
+		// Suhosin(?) or some other strange environment thingy prevents
+		// the direct copy of an array into an index of the same array
+		$copy = $arrayRepresentation;
+		$arrayRepresentation['nodeData'] = $copy;
+
+		if ($this->hasChildNodes() && $addChildNodes) {
+			$arrayRepresentation['children'] = $this->childNodes->toArray();
+		}
+
+		return $arrayRepresentation;
+	}
+
+	/**
+	 * Sets data of the node by a given data array
+	 *
+	 * @param array $data
+	 * @return void
+	 */
+	public function dataFromArray($data) {
+		parent::dataFromArray($data);
+
+		$this->setText($data['label']);
+		$this->setCls($data['cls']);
+		$this->setQTip($data['qtip']);
+		$this->setExpanded($data['expanded']);
+		$this->setExpandable($data['expandable']);
+		$this->setAllowDrag($data['allowDrag']);
+		$this->setAllowDrop($data['isTarget']);
+		$this->setChecked(isset($data['checked']) ? (bool)$data['checked'] : NULL);
+
+		// only set the leaf attribute if it's applied
+		// otherwise you cannot insert nodes into this one
+		if (isset($data['leaf'])) {
+			$this->setLeaf(FALSE);
+		}
+	}
+
+	/**
+	 * @param bool|NULL $checked
+	 */
+	public function setChecked($checked) {
+		$this->checked = $checked;
+	}
+
+	/**
+	 * @return bool|NULL
+	 */
+	public function getChecked() {
+		return $this->checked;
+	}
+
+	/**
+	 * @param int $depth
+	 */
+	public function setDepth($depth) {
+		$this->depth = $depth;
+	}
+
+	/**
+	 * @return int
+	 */
+	public function getDepth() {
+		return $this->depth;
+	}
+
+	/**
+	 * @param NULL|string $href
+	 */
+	public function setHref($href) {
+		$this->href = $href;
+	}
+
+	/**
+	 * @return NULL|string
+	 */
+	public function getHref() {
+		return $this->href;
+	}
+
+	/**
+	 * @param string $hrefTarget
+	 */
+	public function setHrefTarget($hrefTarget) {
+		$this->hrefTarget = $hrefTarget;
+	}
+
+	/**
+	 * @return string
+	 */
+	public function getHrefTarget() {
+		return $this->hrefTarget;
+	}
+
+	/**
+	 * @param string $icon
+	 */
+	public function setIcon($icon) {
+		$this->icon = $icon;
+	}
+
+	/**
+	 * @return string
+	 */
+	public function getIcon() {
+		return $this->icon;
+	}
+
+	/**
+	 * @param string $iconCls
+	 */
+	public function setIconCls($iconCls) {
+		$this->iconCls = $iconCls;
+	}
+
+	/**
+	 * @return string
+	 */
+	public function getIconCls() {
+		return $this->iconCls;
+	}
+
+	/**
+	 * @param int $index
+	 */
+	public function setIndex($index) {
+		$this->index = $index;
+	}
+
+	/**
+	 * @return int
+	 */
+	public function getIndex() {
+		return $this->index;
+	}
+
+	/**
+	 * @param boolean $isFirst
+	 */
+	public function setIsFirst($isFirst) {
+		$this->isFirst = $isFirst;
+	}
+
+	/**
+	 * @return boolean
+	 */
+	public function getIsFirst() {
+		return $this->isFirst;
+	}
+
+	public function setIsLast($isLast) {
+		$this->isLast = $isLast;
+	}
+
+	public function getIsLast() {
+		return $this->isLast;
+	}
+
+
+	/**
+	 * @param string $qtitle
+	 */
+	public function setQtitle($qtitle) {
+		$this->qtitle = $qtitle;
+	}
+
+	/**
+	 * @return string
+	 */
+	public function getQtitle() {
+		return $this->qtitle;
+	}
+
+	/**
+	 * @param boolean $root
+	 */
+	public function setRoot($root) {
+		$this->root = $root;
+	}
+
+	/**
+	 * @return boolean
+	 */
+	public function getRoot() {
+		return $this->root;
+	}
+
+	/**
+	 * returns a string representation
+	 *
+	 * @return string
+	 */
+	public function toString() {
+		return $this->getText();
+	}
+}
+
+if (defined('TYPO3_MODE') && isset($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/extjs/class.t3lib_tree_extjs_node.php'])) {
+	include_once($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/extjs/class.t3lib_tree_extjs_node.php']);
+}
+
+?>
\ No newline at end of file
diff --git a/t3lib/tree/interfaces/interface.t3lib_tree_comparablenode.php b/t3lib/tree/interfaces/interface.t3lib_tree_comparablenode.php
new file mode 100644
index 0000000000000000000000000000000000000000..ed7c0932dc50dc3ccad4babc58fd7c1e15403a54
--- /dev/null
+++ b/t3lib/tree/interfaces/interface.t3lib_tree_comparablenode.php
@@ -0,0 +1,69 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Interface that defines the comparison of nodes
+ *
+ * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+interface t3lib_tree_ComparableNode {
+	/**
+	 * Compare Node against another one
+	 *
+	 * Returns:
+	 * 1 if the current node is greater than the $other,
+	 * -1 if $other is greater than the current node and
+	 * 0 if the nodes are equal
+	 *
+	 * <strong>Example</strong>
+	 * <pre>
+	 *	 if ($this->sortValue > $other->sortValue) {
+	 *		return 1;
+	 *	} elseif ($this->sortValue < $other->sortValue) {
+	 *		return -1;
+	 *	} else {
+	 *		return 0;
+	 *	}
+	 * </pre>
+	 *
+	 * @param t3lib_tree_Node $other
+	 * @return integer see description
+	 */
+	public function compareTo($other);
+
+	/**
+	 * Compares a node if it's identical to another node by the id property.
+	 *
+	 * @param t3lib_tree_Node $other
+	 * @return boolean
+	 */
+	public function equals(t3lib_tree_Node $other);
+
+}
+?>
\ No newline at end of file
diff --git a/t3lib/tree/interfaces/interface.t3lib_tree_draggableanddropable.php b/t3lib/tree/interfaces/interface.t3lib_tree_draggableanddropable.php
new file mode 100644
index 0000000000000000000000000000000000000000..2bdf9cdd4af88778e25059d519f0bf85b79ad4a7
--- /dev/null
+++ b/t3lib/tree/interfaces/interface.t3lib_tree_draggableanddropable.php
@@ -0,0 +1,75 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Describes necessary methods if the nodes are allowDrag and dropable
+ * within the tree.
+ *
+ * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @author Steffen Ritter <info@steffen-ritter.net>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+interface t3lib_tree_DraggableAndDropable {
+	/**
+	 * Moves given node inside a destination node
+	 *
+	 * @param t3lib_tree_Node $node
+	 * @param t3lib_tree_Node $destination
+	 * @return void
+	 */
+	public function moveNodeInDestinationNode($node, $destination);
+
+	/**
+	 * Moves given node after a destination node
+	 *
+	 * @param t3lib_tree_Node $node
+	 * @param t3lib_tree_Node $destination
+	 * @return void
+	 */
+	public function moveNodeAfterDestinationNode($node, $destination);
+
+	/**
+	 * Copies given node inside a destination node
+	 *
+	 * @param t3lib_tree_Node $node
+	 * @param t3lib_tree_Node $destination
+	 * @return void
+	 */
+	public function copyNodeInDestinationNode($node, $destination);
+
+	/**
+	 * Copies given node after a destination node
+	 *
+	 * @param t3lib_tree_Node $node
+	 * @param t3lib_tree_Node $destination
+	 * @return void
+	 */
+	public function copyNodeAfterDestinationNode($node, $destination);
+
+}
+?>
\ No newline at end of file
diff --git a/t3lib/tree/interfaces/interface.t3lib_tree_labeleditable.php b/t3lib/tree/interfaces/interface.t3lib_tree_labeleditable.php
new file mode 100644
index 0000000000000000000000000000000000000000..52f7ce8382c15204ff0fcde1b773289fdfdd7d42
--- /dev/null
+++ b/t3lib/tree/interfaces/interface.t3lib_tree_labeleditable.php
@@ -0,0 +1,46 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Describes necessary methods if the node label should be editable
+ * within the tree.
+ *
+ * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @author Steffen Ritter <info@steffen-ritter.net>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+interface t3lib_tree_LabelEditable {
+	/**
+	 * Sets the new label
+	 *
+	 * @param string $label
+	 * @return void
+	 */
+	public function setLabel($label);
+}
+?>
\ No newline at end of file
diff --git a/t3lib/tree/interfaces/interface.t3lib_tree_recordbasednode.php b/t3lib/tree/interfaces/interface.t3lib_tree_recordbasednode.php
new file mode 100644
index 0000000000000000000000000000000000000000..d4268b78598b9b88b96d9072a888248d403328d4
--- /dev/null
+++ b/t3lib/tree/interfaces/interface.t3lib_tree_recordbasednode.php
@@ -0,0 +1,83 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Interface that defines the nodes based on records
+ *
+ * @author Steffen Ritter <info@steffen-ritter.net>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+interface t3lib_tree_RecordBasedNode {
+
+	/**
+	 * Returns the source field of the label
+	 *
+	 * @return string
+	 */
+	public function getTextSourceField();
+
+	/**
+	 * set the source field of the label
+	 *
+	 * @param string $field
+	 * @return void
+	 */
+	public function setTextSourceField($field);
+
+	/**
+	 * Sets the database record array
+	 *
+	 * @param array $record
+	 * @return void
+	 */
+	public function setRecord($record);
+
+	/**
+	 * Returns the database record array
+	 *
+	 * @return array
+	 */
+	public function getRecord();
+
+	/**
+	 * Returns the table of the record data
+	 *
+	 * @return string
+	 */
+	public function getSourceTable();
+
+	/**
+	 * sets the Table of record source data
+	 *
+	 * @param string $table
+	 * @return void
+	 */
+	public function setSourceTable($table);
+
+}
+?>
diff --git a/t3lib/tree/interfaces/interface.t3lib_tree_renderablenode.php b/t3lib/tree/interfaces/interface.t3lib_tree_renderablenode.php
new file mode 100644
index 0000000000000000000000000000000000000000..1c7ba3e166705371d277aea8ab2969d5fc9a3004
--- /dev/null
+++ b/t3lib/tree/interfaces/interface.t3lib_tree_renderablenode.php
@@ -0,0 +1,50 @@
+<?php
+/***************************************************************
+ *  Copyright notice
+ *
+ *  (c) 2010-2011 TYPO3 Tree Team <http://forge.typo3.org/projects/typo3v4-extjstrees>
+ *  All rights reserved
+ *
+ *  This script is part of the TYPO3 project. The TYPO3 project is
+ *  free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  The GNU General Public License can be found at
+ *  http://www.gnu.org/copyleft/gpl.html.
+ *  A copy is found in the textfile GPL.txt and important notices to the license
+ *  from the author is found in LICENSE.txt distributed with these scripts.
+ *
+ *
+ *  This script is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  This copyright notice MUST APPEAR in all copies of the script!
+ ***************************************************************/
+
+/**
+ * Interface that defines the to be rendered
+ *
+ * @author Steffen Ritter <info@steffen-ritter.net>
+ * @package TYPO3
+ * @subpackage t3lib
+ */
+interface t3lib_tree_RenderableNode {
+	/**
+	 * returns a string representation
+	 *
+	 * @return string
+	 */
+	public function toString();
+
+	/**
+	 * returns an array representation of an node
+	 *
+	 * @return array
+	 */
+	public function toArray();
+}
+?>
diff --git a/t3lib/tree/extdirect/class.t3lib_tree_extdirect_node.php b/t3lib/tree/navigationtree/class.t3lib_tree_navigationtree_node.php
similarity index 65%
rename from t3lib/tree/extdirect/class.t3lib_tree_extdirect_node.php
rename to t3lib/tree/navigationtree/class.t3lib_tree_navigationtree_node.php
index 625ad0028aac54243f09c9bbc9c5f66cd012d169..678aaa62493c6bd7517d3971bba5c35f183b34f1 100644
--- a/t3lib/tree/extdirect/class.t3lib_tree_extdirect_node.php
+++ b/t3lib/tree/navigationtree/class.t3lib_tree_navigationtree_node.php
@@ -26,68 +26,37 @@
 ***************************************************************/
 
 /**
- * Node for the usage with ExtDirect and ExtJS
+ * Extends the ExtJS-Tree Nodes with custom features implemented for the TYPO3 Navigation Panel
  *
- * @author Stefan Galinski <stefan.galinski@gmail.com>
+ * @author Steffen Ritter <info@steffen-ritter.net>
  * @package TYPO3
  * @subpackage t3lib
  */
-class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
-	/**
-	 * Node type
-	 *
-	 * @var string
-	 */
-	protected $type = '';
+class t3lib_tree_NavigationTree_Node extends t3lib_tree_ExtJs_Node {
 
-	/**
-	 * Leaf Node Indicator
-	 *
-	 * @var bool
-	 */
-	protected $leaf = TRUE;
 
 	/**
-	 * Indicator if the node is expanded
-	 *
-	 * @var bool
-	 */
-	protected $expanded = FALSE;
-
-	/**
-	 * Indicator if the node can be expanded
+	 * Database record (not serialized or merged into the result array!)
 	 *
-	 * @var bool
+	 * @var array
 	 */
-	protected $expandable = FALSE;
+	protected $record = array();
 
-	/**
-	 * Indicator if the node is draggable
-	 *
-	 * @var bool
-	 */
-	protected $draggable = TRUE;
 
 	/**
-	 * Indicator if the node is allowed as a drop target
+	 * Indicator if the node can have children's
 	 *
 	 * @var bool
 	 */
-	protected $isDropTarget = TRUE;
+	protected $allowChildren = TRUE;
 
 	/**
-	 * Label
+	 * Node type
 	 *
 	 * @var string
 	 */
-	protected $text = '';
+	protected $type = '';
 
-	/**
-	 * Editable Label text
-	 *
-	 * @var string
-	 */
-	protected $editableText = '';
 
 	/**
 	 * Prefix text of the label
@@ -104,60 +73,19 @@ class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
 	protected $suffix = '';
 
 	/**
-	 * CSS Class
-	 *
-	 * @var string
-	 */
-	protected $cls = '';
-
-	/**
-	 * Quick Tip
-	 *
-	 * @var string
-	 */
-	protected $qtip = '';
-
-	/**
-	 * Sprite Icon HTML
+	 * Context Info
 	 *
-	 * @var string
+	 * @var array
 	 */
-	protected $spriteIconCode = '';
+	protected $contextInfo = array();
 
 	/**
-	 * Text source field (title, nav_title, ...)
+	 * Editable Label text
 	 *
 	 * @var string
 	 */
-	protected $t3TextSourceField = '';
-
-	/**
-	 * Indicator if the copy mode is activated
-	 *
-	 * @var bool
-	 */
-	protected $t3InCopyMode = FALSE;
-
-	/**
-	 * Indicator if the cut mode is activated
-	 *
-	 * @var bool
-	 */
-	protected $t3InCutMode = FALSE;
-
-	/**
-	 * Database record (not serialized or merged into the result array!)
-	 *
-	 * @var array
-	 */
-	protected $record = array();
+	protected $editableText = '';
 
-	/**
-	 * Context Info
-	 *
-	 * @var array
-	 */
-	protected $contextInfo = array();
 
 	/**
 	 * Indicator if the label is editable
@@ -166,173 +94,92 @@ class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
 	 */
 	protected $labelIsEditable = TRUE;
 
-	/**
-	 * Indicator if the node can have children's
-	 *
-	 * @var bool
-	 */
-	protected $allowChildren = TRUE;
 
 	/**
-	 * Set's the node type
-	 *
-	 * @param string $type
-	 * @return void
-	 */
-	public function setType($type) {
-		$this->type = $type;
-	}
-
-	/**
-	 * Returns the node type
-	 *
-	 * @return string
-	 */
-	public function getType() {
-		return $this->type;
-	}
-
-	/**
-	 * Sets the leaf node indicator
+	 * Sets the context info
 	 *
-	 * @param bool $isLeaf
+	 * @param array $contextInfo
 	 * @return void
 	 */
-	public function setLeaf($isLeaf) {
-		$this->leaf = ($isLeaf == TRUE);
-	}
-
-	/**
-	 * Returns if the node is a leaf node
-	 *
-	 * @return bool
-	 */
-	public function isLeafNode() {
-		return $this->leaf;
+	public function setContextInfo($contextInfo) {
+		$this->contextInfo = (array) $contextInfo;
 	}
 
 	/**
-	 * Sets the expandable indicator
+	 * Returns the context info
 	 *
-	 * @param bool $expandable
-	 * @return void
+	 * @return array
 	 */
-	public function setExpandable($expandable) {
-		$this->expandable = ($expandable == TRUE);
+	public function getContextInfo() {
+		return (array) $this->contextInfo;
 	}
 
-	/**
-	 * Returns the expandable indicator
-	 *
-	 * @return bool
-	 */
-	public function isExpandable() {
-		return $this->expandable;
-	}
 
 	/**
-	 * Sets the expanded indicator
+	 * Sets the indicator if the node can have child nodes
 	 *
-	 * @param bool $expanded
+	 * @param boolean $allowChildren
 	 * @return void
 	 */
-	public function setExpanded($expanded) {
-		$this->expanded = ($expanded == TRUE);
+	public function setAllowChildren($allowChildren) {
+		$this->allowChildren = ($allowChildren == TRUE);
 	}
 
 	/**
-	 * Returns the expanded indicator
+	 * Checks if the node can have child nodes
 	 *
 	 * @return bool
 	 */
-	public function isExpanded() {
-		if ($this->isLeafNode()) {
-			return TRUE;
-		}
-
-		return $this->expanded;
-	}
-
-	/**
-	 * Sets the draggable indicator
-	 *
-	 * @param bool $draggable
-	 * @return void
-	 */
-	public function setDraggable($draggable) {
-		$this->draggable = ($draggable == TRUE);
+	public function canHaveChildren() {
+		return $this->allowChildren;
 	}
 
-	/**
-	 * Returns the draggable indicator
-	 *
-	 * @return bool
-	 */
-	public function isDraggable() {
-		return $this->draggable;
-	}
 
 	/**
-	 * Sets the indicator if the node can be a drop target
+	 * Sprite Icon HTML
 	 *
-	 * @param bool $isDropTarget
-	 * @return void
+	 * @var string
 	 */
-	public function setIsDropTarget($isDropTarget) {
-		$this->isDropTarget = ($isDropTarget == TRUE);
-	}
+	protected $spriteIconCode = '';
 
 	/**
-	 * Returns the indicator if the node is a drop target
+	 * Text source field (title, nav_title, ...)
 	 *
-	 * @return bool
+	 * @var string
 	 */
-	public function isDropTarget() {
-		return $this->isDropTarget;
-	}
+	protected $t3TextSourceField = '';
 
 	/**
-	 * Sets the label of the node with the source field and the prefix
+	 * Indicator if the copy mode is activated
 	 *
-	 * @param string $text
-	 * @param string $textSourceField
-	 * @param string $prefix
-	 * @param string $suffix
-	 * @return void
+	 * @var bool
 	 */
-	public function setText($text, $textSourceField = 'title', $prefix = '', $suffix = '') {
-		$this->text = $text;
-		$this->t3TextSourceField = $textSourceField;
-		$this->prefix = $prefix;
-		$this->suffix = $suffix;
-	}
+	protected $t3InCopyMode = FALSE;
 
 	/**
-	 * Returns the label
+	 * Indicator if the cut mode is activated
 	 *
-	 * @return string
+	 * @var bool
 	 */
-	public function getText() {
-		return $this->text;
-	}
+	protected $t3InCutMode = FALSE;
 
-	/**
-	 * Sets the editable text
+		/**
+	 * Sets the sprite icon code
 	 *
-	 * @param string $editableText
+	 * @param string $spriteIcon
 	 * @return void
 	 */
-	public function setEditableText($editableText) {
-		$this->editableText = $editableText;
+	public function setSpriteIconCode($spriteIcon) {
+		$this->spriteIconCode = $spriteIcon;
 	}
 
 	/**
-	 * Returns the editable text
+	 * Returns the sprite icon code
 	 *
 	 * @return string
 	 */
-	public function getEditableText() {
-		return $this->editableText;
+	public function getSpriteIconCode() {
+		return $this->spriteIconCode;
 	}
 
 	/**
@@ -399,63 +246,45 @@ class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
 	public function getSuffix() {
 		return $this->suffix;
 	}
-
+	
 	/**
-	 * Sets the css class(es)
+	 * Set's the node type
 	 *
-	 * @param string $class
+	 * @param string $type
 	 * @return void
 	 */
-	public function setCls($class) {
-		$this->cls = $class;
+	public function setType($type) {
+		$this->type = $type;
 	}
 
 	/**
-	 * Returns the css class(es)
+	 * Returns the node type
 	 *
 	 * @return string
 	 */
-	public function getCls() {
-		return $this->cls;
+	public function getType() {
+		return $this->type;
 	}
 
 	/**
-	 * Sets the quick tip
+	 * Sets the editable text
 	 *
-	 * @param string $qtip
+	 * @param string $editableText
 	 * @return void
 	 */
-	public function setQTip($qtip) {
-		$this->qtip = $qtip;
+	public function setEditableText($editableText) {
+		$this->editableText = $editableText;
 	}
 
 	/**
-	 * Returns the quick tip
+	 * Returns the editable text
 	 *
 	 * @return string
 	 */
-	public function getQTip() {
-		return $this->qtip;
-	}
-
-	/**
-	 * Sets the sprite icon code
-	 *
-	 * @param string $spriteIcon
-	 * @return void
-	 */
-	public function setSpriteIconCode($spriteIcon) {
-		$this->spriteIconCode = $spriteIcon;
+	public function getEditableText() {
+		return $this->editableText;
 	}
 
-	/**
-	 * Returns the sprite icon code
-	 *
-	 * @return string
-	 */
-	public function getSpriteIconCode() {
-		return $this->spriteIconCode;
-	}
 
 	/**
 	 * Sets the indicator if the label is editable
@@ -477,74 +306,39 @@ class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
 	}
 
 	/**
-	 * Sets the database record array
-	 *
-	 * @param array $record
-	 * @return void
-	 */
-	public function setRecord($record) {
-		$this->record = (array) $record;
-	}
-
-	/**
-	 * Returns the database record array
-	 *
-	 * @return array
-	 */
-	public function getRecord() {
-		return $this->record;
-	}
-
-	/**
-	 * Sets the context info
-	 *
-	 * @param array $contextInfo
-	 * @return void
-	 */
-	public function setContextInfo($contextInfo) {
-		$this->contextInfo = (array) $contextInfo;
-	}
-
-	/**
-	 * Returns the context info
-	 *
-	 * @return array
-	 */
-	public function getContextInfo() {
-		return (array) $this->contextInfo;
-	}
-
-	/**
-	 * Sets the child nodes collection
+	 * Sets the label of the node with the source field and the prefix
 	 *
-	 * @param t3lib_tree_NodeCollection $childNodes
+	 * @param string $text
+	 * @param string $textSourceField
+	 * @param string $prefix
+	 * @param string $suffix
 	 * @return void
 	 */
-	public function setChildNodes(t3lib_tree_NodeCollection $childNodes) {
-		parent::setChildNodes($childNodes);
-
-		if ($childNodes->count()) {
-			$this->setLeaf(FALSE);
-		}
+	public function setText($text, $textSourceField = 'title', $prefix = '', $suffix = '') {
+		$this->text = $text;
+		$this->t3TextSourceField = $textSourceField;
+		$this->prefix = $prefix;
+		$this->suffix = $suffix;
 	}
 
 	/**
-	 * Sets the indicator if the node can have child nodes
+	 * Sets data of the node by a given data array
 	 *
-	 * @param boolean $allowChildren
+	 * @param array $data
 	 * @return void
 	 */
-	public function setAllowChildren($allowChildren) {
-		$this->allowChildren = ($allowChildren == TRUE);
-	}
+	public function dataFromArray($data) {
+		parent::dataFromArray($data);
 
-	/**
-	 * Checks if the node can have child nodes
-	 *
-	 * @return bool
-	 */
-	public function canHaveChildren() {
-		return $this->allowChildren;
+		$this->setText($data['label'], $data['t3TextSourceField'], $data['prefix'], $data['suffix']);
+		$this->setType($data['type']);
+		$this->setEditableText($data['editableText']);
+		$this->setSpriteIconCode($data['spriteIconCode']);
+		$this->setInCopyMode($data['t3InCopyMode']);
+		$this->setInCutMode($data['t3InCutMode']);
+		$this->setContextInfo($data['t3ContextInfo']);
+		$this->setLabelIsEditable($data['editable']);
+		$this->setAllowChildren($data['allowChildren']);
 	}
 
 	/**
@@ -554,20 +348,14 @@ class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
 	 * @return array
 	 */
 	public function toArray($addChildNodes = TRUE) {
-		$arrayRepresentation = array(
-			'serializeClassName' => get_class($this),
-			'id' => $this->getId(),
+		$arrayRepresentation = parent::toArray($addChildNodes);
+
+		$arrayRepresentation2 = array(
 			'type' => $this->getType(),
 			'editableText' => $this->getEditableText(),
 			'text' => $this->getPrefix() . $this->getText() . $this->getSuffix(),
-			'cls' => $this->getCls(),
 			'prefix' => $this->getPrefix(),
 			'suffix' => $this->getSuffix(),
-			'qtip' => $this->getQTip(),
-			'expanded' => $this->isExpanded(),
-			'expandable' => $this->isExpandable(),
-			'draggable' => $this->isDraggable(),
-			'isTarget' => $this->isDropTarget(),
 			'spriteIconCode' => $this->getSpriteIconCode(),
 			't3TextSourceField' => $this->getTextSourceField(),
 			't3InCopyMode' => $this->isInCopyMode(),
@@ -580,56 +368,51 @@ class t3lib_tree_extdirect_Node extends t3lib_tree_Node {
 			// only set the leaf attribute if the node has children's,
 			// otherwise you cannot add child's to real leaf nodes
 		if (!$this->isLeafNode()) {
-			$arrayRepresentation['leaf'] = FALSE;
+			$arrayRepresentation2['leaf'] = FALSE;
 		}
 
+			// Do not nest nodeData and children arrays
+		unset($arrayRepresentation['nodeData']);
+		unset($arrayRepresentation['children']);
+		$arrayRepresentation = array_merge($arrayRepresentation, $arrayRepresentation2);
+
 			// Suhosin(?) or some other strange environment thingy prevents
 			// the direct copy of an array into an index of the same array
 		$copy = $arrayRepresentation;
 		$arrayRepresentation['nodeData'] = $copy;
 
-		if ($this->hasChildNodes()) {
+		if ($this->hasChildNodes() && $addChildNodes) {
 			$arrayRepresentation['children'] = $this->childNodes->toArray();
 		}
 
 		return $arrayRepresentation;
 	}
 
+
 	/**
-	 * Sets data of the node by a given data array
+	 * Sets the database record array
 	 *
-	 * @param array $data
+	 * @param array $record
 	 * @return void
 	 */
-	public function dataFromArray($data) {
-		parent::dataFromArray($data);
-
-		$this->setType($data['type']);
-		$this->setText($data['label'], $data['t3TextSourceField'], $data['prefix'], $data['suffix']);
-		$this->setEditableText($data['editableText']);
-		$this->setCls($data['cls']);
-		$this->setQTip($data['qtip']);
-		$this->setExpanded($data['expanded']);
-		$this->setExpandable($data['expandable']);
-		$this->setDraggable($data['draggable']);
-		$this->setIsDropTarget($data['isTarget']);
-		$this->setSpriteIconCode($data['spriteIconCode']);
-		$this->setInCopyMode($data['t3InCopyMode']);
-		$this->setInCutMode($data['t3InCutMode']);
-		$this->setContextInfo($data['t3ContextInfo']);
-		$this->setLabelIsEditable($data['editable']);
-		$this->setAllowChildren($data['allowChildren']);
+	public function setRecord($record) {
+		$this->record = (array) $record;
+	}
 
-			// only set the leaf attribute if it's applied
-			// otherwise you cannot insert nodes into this one
-		if (isset($data['leaf'])) {
-			$this->setLeaf(FALSE);
-		}
+	/**
+	 * Returns the database record array
+	 *
+	 * @return array
+	 */
+	public function getRecord() {
+		return $this->record;
 	}
+
+
 }
 
-if (defined('TYPO3_MODE') && isset($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/extdirect/class.t3lib_tree_extdirect_node.php'])) {
-	include_once($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/extdirect/class.t3lib_tree_extdirect_node.php']);
+if (defined('TYPO3_MODE') && isset($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/navigationtree/class.t3lib_tree_navigationtree_node.php'])) {
+	include_once($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/navigationtree/class.t3lib_tree_navigationtree_node.php']);
 }
 
 ?>
\ No newline at end of file
diff --git a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_commands.php b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_commands.php
index 532761ede0fca3788b6455f6c8db3b23749af1f8..782c7b5609c51db8a8bf417f75213f2026b98774 100644
--- a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_commands.php
+++ b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_commands.php
@@ -36,7 +36,7 @@ final class t3lib_tree_pagetree_Commands {
 	/**
 	 * Visibly the page
 	 *
-	 * @param t3lib_tree_pagetree_Node $nodeData
+	 * @param t3lib_tree_pagetree_Node $node
 	 * @return void
 	 */
 	public static function visiblyNode(t3lib_tree_pagetree_Node $node) {
@@ -47,7 +47,7 @@ final class t3lib_tree_pagetree_Commands {
 	/**
 	 * Hide the page
 	 *
-	 * @param t3lib_tree_pagetree_Node $nodeData
+	 * @param t3lib_tree_pagetree_Node $node
 	 * @return void
 	 */
 	public static function disableNode(t3lib_tree_pagetree_Node $node) {
@@ -58,7 +58,7 @@ final class t3lib_tree_pagetree_Commands {
 	/**
 	 * Delete the page
 	 *
-	 * @param t3lib_tree_pagetree_Node $nodeData
+	 * @param t3lib_tree_pagetree_Node $node
 	 * @return void
 	 */
 	public static function deleteNode(t3lib_tree_pagetree_Node $node) {
@@ -69,7 +69,7 @@ final class t3lib_tree_pagetree_Commands {
 	/**
 	 * Restore the page
 	 *
-	 * @param t3lib_tree_pagetree_Node $nodeData
+	 * @param t3lib_tree_pagetree_Node $node
 	 * @param int $targetId
 	 * @return void
 	 */
@@ -85,7 +85,7 @@ final class t3lib_tree_pagetree_Commands {
 	/**
 	 * Updates the node label
 	 *
-	 * @param t3lib_tree_pagetree_Node $nodeData
+	 * @param t3lib_tree_pagetree_Node $node
 	 * @param string $updatedLabel
 	 * @return void
 	 */
@@ -276,6 +276,7 @@ final class t3lib_tree_pagetree_Commands {
 	 * Creates a node with the given record information's
 	 *
 	 * @param array $record
+	 * @param int $mountPoint
 	 * @return t3lib_tree_pagetree_Node
 	 */
 	public static function getNewNode($record, $mountPoint = 0) {
@@ -335,7 +336,7 @@ final class t3lib_tree_pagetree_Commands {
 		if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['GLOBAL']['recStatInfoHooks'])) {
 			$_params = array('pages', $record['uid']);
 			foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['GLOBAL']['recStatInfoHooks'] as $_funcRef) {
-				$stat .= t3lib_div::callUserFunction($_funcRef, $_params, $this);
+				$stat .= t3lib_div::callUserFunction($_funcRef, $_params, NULL);
 			}
 		}
 
@@ -358,11 +359,11 @@ final class t3lib_tree_pagetree_Commands {
 		$subNode->setSpriteIconCode($spriteIconCode);
 
 		if (!$subNode->canCreateNewPages() || intval($record['t3ver_state']) === 2) {
-			$subNode->setIsDropTarget(FALSE);
+			$subNode->setAllowDrop(FALSE);
 		}
 
 		if (!$subNode->canBeEdited() || !$subNode->canBeRemoved() || intval($record['t3ver_state']) === 2) {
-			$subNode->setDraggable(FALSE);
+			$subNode->setAllowDrag(FALSE);
 		}
 
 		return $subNode;
diff --git a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_dataprovider.php b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_dataprovider.php
index 4cd6f238992702ad32f5b53cf9b7d04c2394728c..78b010ddd7be109b1b721badd48da3e1663c7167 100644
--- a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_dataprovider.php
+++ b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_dataprovider.php
@@ -123,7 +123,7 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 		if (!is_array($subpages) || !count($subpages)) {
 			return $nodeCollection;
 		}
-
+		$index = -1;
 		foreach ($subpages as $subpage) {
 			if (in_array($subpage['uid'], $this->hiddenRecords)) {
 				continue;
@@ -135,14 +135,20 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 			}
 
 			$subNode = t3lib_tree_pagetree_Commands::getNewNode($subpage, $mountPoint);
+			$subNode->setParentNode($node);
+			$subNode->setDepth($node->getDepth() + 1);
+			$subNode->setIndex(++$index);
+			$subNode->setIsFirst($index === 0);
+			$subNode->setIsLast($index === (count($subpages)-1));
 			if ($this->nodeCounter < $this->nodeLimit) {
 				$childNodes = $this->getNodes($subNode, $mountPoint, $level + 1);
 				$subNode->setChildNodes($childNodes);
+				$subNode->setLeaf(!$subNode->hasChildNodes());
 				$this->nodeCounter += $childNodes->count();
 			} else {
 				$subNode->setLeaf(!$this->hasNodeSubPages($subNode->getId()));
 			}
-
+			$subNode->setExpandable(!$subNode->isLeafNode());
 			$nodeCollection->append($subNode);
 		}
 
@@ -160,7 +166,7 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 	 * @param t3lib_tree_Node $node
 	 * @param string $searchFilter
 	 * @param int $mountPoint
-	 * @return void
+	 * @return t3lib_tree_pagetree_NodeCollection
 	 */
 	public function getFilteredNodes(t3lib_tree_Node $node, $searchFilter, $mountPoint = 0) {
 		/** @var $nodeCollection t3lib_tree_pagetree_NodeCollection */
@@ -205,13 +211,10 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 				if ($reference && $reference->offsetExists($ident)) {
 					/** @var $refNode t3lib_tree_pagetree_Node */
 					$refNode = $reference->offsetGet($ident);
-					$refNode->setExpanded(TRUE);
-					$refNode->setLeaf(FALSE);
-
-					$reference = $refNode->getChildNodes();
-					if ($reference == NULL) {
-						$reference = t3lib_div::makeInstance('t3lib_tree_pagetree_NodeCollection');
-						$refNode->setChildNodes($reference);
+					if ($refNode->hasChildNodes()) {
+						$refNode->setExpanded(TRUE);
+						$refNode->setLeaf(FALSE);
+						$reference = $refNode->getChildNodes();
 					}
 				} else {
 					$refNode = t3lib_tree_pagetree_Commands::getNewNode($rootlineElement, $mountPoint);
@@ -287,6 +290,7 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 		}
 
 		$showRootlineAboveMounts = $GLOBALS['BE_USER']->getTSConfigVal('options.pageTree.showPathAboveMounts');
+		$index = -1;
 		foreach ($mountPoints as $mountPoint) {
 			if ($mountPoint === 0) {
 				$sitename = 'TYPO3';
@@ -322,10 +326,13 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 				$subNode->setExpanded(TRUE);
 				$subNode->setCls('typo3-pagetree-node-notExpandable');
 			}
-
+			$subNode->setDepth(1);
+			$subNode->setIndex(++$index);
+			$subNode->setIsFirst($index === 0);
+			$subNode->setIsLast($index === (count($mountPoints)-1));
 			$subNode->setIsMountPoint(TRUE);
-			$subNode->setDraggable(FALSE);
-			$subNode->setIsDropTarget(FALSE);
+			$subNode->setAllowDrag(FALSE);
+			$subNode->setAllowDrop(FALSE);
 
 			if ($searchFilter === '') {
 				$childNodes = $this->getNodes($subNode, $mountPoint);
@@ -363,21 +370,22 @@ class t3lib_tree_pagetree_DataProvider extends t3lib_tree_AbstractDataProvider {
 		}
 
 		if ($searchFilter !== '') {
+			$searchWhere = '';
 			if (is_numeric($searchFilter) && $searchFilter > 0) {
-				$seachWhere .= 'uid = ' . intval($searchFilter) . ' OR ';
+				$searchWhere .= 'uid = ' . intval($searchFilter) . ' OR ';
 			}
 
 			$searchFilter = $GLOBALS['TYPO3_DB']->fullQuoteStr('%' . $searchFilter . '%', 'pages');
 			$useNavTitle = $GLOBALS['BE_USER']->getTSConfigVal('options.pageTree.showNavTitle');
 
 			if ($useNavTitle) {
-				$seachWhere .= '(nav_title LIKE ' . $searchFilter .
+				$searchWhere .= '(nav_title LIKE ' . $searchFilter .
 					' OR (nav_title = "" && title LIKE ' . $searchFilter . '))';
 			} else {
-				$seachWhere .= 'title LIKE ' . $searchFilter;
+				$searchWhere .= 'title LIKE ' . $searchFilter;
 			}
 
-			$where .= ' AND (' . $seachWhere . ')';
+			$where .= ' AND (' . $searchWhere . ')';
 		}
 
 		return $where;
diff --git a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_indicator.php b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_indicator.php
index a97f7ac826996a4b8ece8f81fbc4072c1b889e6a..619273a2b9185ef3e2473cfebceb54a245400db4 100644
--- a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_indicator.php
+++ b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_indicator.php
@@ -53,7 +53,7 @@ class t3lib_tree_pagetree_Indicator {
 	 */
 	protected function getIndicatorProviders() {
 		$providers = $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['pagetree']['t3lib_tree_pagetree']['indicator']['providers'];
-		if (!is_array($providers)) {
+		if (!is_array($providers) || count($this->indicatorProviders) > 0) {
 			return;
 		}
 
@@ -69,13 +69,13 @@ class t3lib_tree_pagetree_Indicator {
 	/**
 	 * Runs through all indicator providers and returns all indicators collected.
 	 *
-	 * @return	array	An array of
+	 * @return	array	An array of indicator objects
 	 */
 	public function getAllIndicators() {
 		$indicators = array();
 		foreach ($this->indicatorProviders as $indicatorProvider) {
 			$indicator = $indicatorProvider->getIndicator();
-			if($indicator) {
+			if ($indicator) {
 				$indicators[] = $indicator;
 			}
 		}
diff --git a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_node.php b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_node.php
index 91af17c4d33001ee575bf57ad57be14f6554b160..4c5145933a583b76d453a47d2018ebe170a7eb4c 100644
--- a/t3lib/tree/pagetree/class.t3lib_tree_pagetree_node.php
+++ b/t3lib/tree/pagetree/class.t3lib_tree_pagetree_node.php
@@ -32,7 +32,7 @@
  * @package TYPO3
  * @subpackage t3lib
  */
-class t3lib_tree_pagetree_Node extends t3lib_tree_extdirect_Node {
+class t3lib_tree_pagetree_Node extends t3lib_tree_NavigationTree_Node implements t3lib_tree_RecordBasedNode {
 	/**
 	 * Cached access rights to save some performance
 	 *
@@ -297,6 +297,19 @@ class t3lib_tree_pagetree_Node extends t3lib_tree_extdirect_Node {
 	public function canBeTemporaryMountPoint() {
 		return TRUE;
 	}
+	
+	/**
+	 * Returns a special id for a page tree node
+	 *
+	 * @return string
+	 */
+	public function getPageTreeNodeId() {
+		if ($this->getId() === 'root') {
+			return 'root';
+		} else {
+			return 'p' . dechex($this->getId()) . ($this->getMountPoint() ? '-' . dechex($this->getMountPoint()) : '');
+		}
+	}
 
 	/**
 	 * Returns the node in an array representation that can be used for serialization
@@ -307,17 +320,34 @@ class t3lib_tree_pagetree_Node extends t3lib_tree_extdirect_Node {
 	public function toArray($addChildNodes = TRUE) {
 		$arrayRepresentation = parent::toArray();
 
-		$arrayRepresentation['id'] = 'p' . dechex($this->getId()) . ($this->getMountPoint() ? '-' . dechex($this->getMountPoint()) : '');
-		$arrayRepresentation['realId'] = $this->getId();
-		$arrayRepresentation['nodeData']['id'] = $this->getId();
-
-		$arrayRepresentation['readableRootline'] = $this->getReadableRootline();
-		$arrayRepresentation['nodeData']['readableRootline'] = $this->getReadableRootline();
-
-		$arrayRepresentation['nodeData']['mountPoint'] = $this->getMountPoint();
-		$arrayRepresentation['nodeData']['workspaceId'] = $this->getWorkspaceId();
-		$arrayRepresentation['nodeData']['isMountPoint'] = $this->isMountPoint();
-		$arrayRepresentation['nodeData']['serializeClassName'] = get_class($this);
+		$arrayRepresentation2['id'] = $this->getPageTreeNodeId();
+		$arrayRepresentation2['realId'] = $this->getId();
+		$arrayRepresentation2['serializeClassName'] = get_class($this);
+		$arrayRepresentation2['parentId'] = (!$this->getRoot() && $this->getParentNode()) ? $this->getParentNode()->getPageTreeNodeId() : 'root';
+		$arrayRepresentation2['readableRootline'] = $this->getReadableRootline();
+		$arrayRepresentation2['mountPoint'] = $this->getMountPoint();
+		$arrayRepresentation2['workspaceId'] = $this->getWorkspaceId();
+		$arrayRepresentation2['isMountPoint'] = $this->isMountPoint();
+
+			// only set the leaf attribute if the node has children's,
+			// otherwise you cannot add child's to real leaf nodes
+		if (!$this->isLeafNode()) {
+			$arrayRepresentation2['leaf'] = FALSE;
+		}
+			// Do not nest nodeData and children arrays
+		unset($arrayRepresentation['nodeData']);
+		unset($arrayRepresentation['children']);
+		$arrayRepresentation = array_merge($arrayRepresentation, $arrayRepresentation2);
+
+			// Suhosin(?) or some other strange environment thingy prevents
+			// the direct copy of an array into an index of the same array
+		$copy = $arrayRepresentation;
+		$arrayRepresentation['nodeData'] = $copy;
+		$arrayRepresentation['nodeData']['id'] = $arrayRepresentation['realId'];
+
+		if ($this->hasChildNodes() && $addChildNodes) {
+			$arrayRepresentation['children'] = $this->childNodes->toArray();
+		}
 
 		return $arrayRepresentation;
 	}
@@ -335,6 +365,35 @@ class t3lib_tree_pagetree_Node extends t3lib_tree_extdirect_Node {
 		$this->setReadableRootline($data['readableRootline']);
 		$this->setIsMountPoint($data['isMountPoint']);
 	}
+
+	/**
+	 * set the source field of the label
+	 *
+	 * @param string $field
+	 * @return void
+	 */
+	public function setTextSourceField($field) {
+		$this->t3TextSourceField = $field;
+	}
+
+	/**
+	 * Returns the table of the record data
+	 *
+	 * @return string
+	 */
+	public function getSourceTable() {
+		return 'pages';
+	}
+
+	/**
+	 * sets the Table of record source data
+	 *
+	 * @param string $table
+	 * @return void
+	 */
+	public function setSourceTable($table) {
+		// not needed, pagetree always is 'pages'
+	}
 }
 
 if (defined('TYPO3_MODE') && isset($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['t3lib/tree/pagetree/class.t3lib_tree_pagetree_node.php'])) {
diff --git a/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_commands.php b/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_commands.php
index a3a33dac362ca8b8d76c4abedbb55fe3840f83cc..12ccaeb8ed014d648c3b63670e664ea81d20f6f4 100644
--- a/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_commands.php
+++ b/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_commands.php
@@ -225,7 +225,7 @@ class t3lib_tree_pagetree_extdirect_Commands {
 	 *
 	 * @param stdClass $nodeData
 	 * @param int $destination
-	 * @return void
+	 * @return array
 	 */
 	public function moveNodeAfterDestination($nodeData, $destination) {
 		/** @var $node t3lib_tree_pagetree_Node */
diff --git a/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_tree.php b/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_tree.php
index e8df3f8f2ab2711fd799603347925d0aa5cb6c9f..466195f9c6763f6794e0a5555c744b2fda98729f 100644
--- a/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_tree.php
+++ b/t3lib/tree/pagetree/extdirect/class.t3lib_tree_pagetree_extdirect_tree.php
@@ -32,7 +32,7 @@
  * @package TYPO3
  * @subpackage t3lib
  */
-class t3lib_tree_pagetree_extdirect_Tree extends t3lib_tree_ExtDirect_AbstractExtJsTree {
+class t3lib_tree_pagetree_extdirect_Tree extends t3lib_tree_ExtJs_AbstractStatefulExtJsTree {
 	/**
 	 * Data Provider
 	 *
@@ -180,7 +180,7 @@ class t3lib_tree_pagetree_extdirect_Tree extends t3lib_tree_ExtDirect_AbstractEx
 	/**
 	 * Returns the language labels, sprites and configuration options for the pagetree
 	 *
-	 * @return void
+	 * @return array
 	 */
 	public function loadResources() {
 		$file = 'LLL:EXT:lang/locallang_core.xml:';
diff --git a/typo3/sysext/t3skin/stylesheets/structure/element_pagetree.css b/typo3/sysext/t3skin/stylesheets/structure/element_pagetree.css
index 8ae4e364b21ce822b27a23892984d893099f3695..aed04b76ab3eca17b7f9cf0be8a42968c2c8e6fb 100644
--- a/typo3/sysext/t3skin/stylesheets/structure/element_pagetree.css
+++ b/typo3/sysext/t3skin/stylesheets/structure/element_pagetree.css
@@ -6,50 +6,32 @@ Pagetree
 #typo3-pagetree,
 #typo3-pagetree .x-panel-bwrap,
 #typo3-pagetree .x-panel-body {
+	border: none;
 	height: 100%;
 }
 
-#typo3-pagetree .x-panel-tbar {
-	padding: 0;
-	margin: 0;
-}
-
-.x-tree-node .x-tree-node-el {
-	margin-right: 0;
+.typo3-pagetree .x-grid-table {
+	padding-left: 5px;
 }
 
 .x-tree-node-readableRootline {
 	padding: 10px 0 3px 10px;
 }
 
-.x-tree-lines .typo3-pagetree-node-notExpandable .x-tree-ec-icon {
-	visibility: hidden;
-}
-
-.x-tree-lines .typo3-pagetree-node-notExpandable ul .x-tree-ec-icon {
-	visibility: visible;
-}
-
-#typo3-pagetree span.t3-icon {
+.typo3-pagetree span.t3-icon {
 	margin-bottom: 2px;
 }
 
-#typo3-pagetree .x-toolbar,
-.typo3-pagetree-topPanel-item .x-toolbar {
-	padding-bottom: 0;
-}
-
-#typo3-pagetree ul.x-tree-root-ct {
-	padding-left: 5px;
-	padding-top: 15px;
-}
-
-.x-tree-node a .typo3-pagetree-status,
+.typo3-pagetree .x-grid-row .x-grid-cell a .typo3-pagetree-status,
 .x-dd-drag-ghost a .typo3-pagetree-status {
 	padding: 0;
 	margin: 0 3px 0 0;
 }
 
+.typo3-pagetree .x-form-field {
+	height: 19px;
+}
+
 /**
  *
  * section drag and drop
@@ -86,62 +68,62 @@ Pagetree
  * @section Top Panel
  *
  */
-#typo3-pagetree-topPanel .x-toolbar-left {
-	height: 20px;
+#typo3-pagetree .x-docked-top {
+	padding: 0;
 }
 
-.typo3-pagetree-topPanel-button {
-	margin: 0 5px 0 0;
-	padding: 1px 2px;
-	height: 18px;
+.typo3-pagetree-topPanel {
+	height: 49px;
 }
 
-.typo3-pagetree-topPanel-button button {
-	height: 16px;
-	width: 16px;
+#typo3-pagetree-topPanel .x-docked-top .x-box-inner {
+	height: 22px;
+	margin: 0;
 }
 
-.typo3-pagetree-topPanel-item {
-	padding: 5px 3px 0 3px;
-	height: 22px;
+#typo3-pagetree .typo3-pagetree-topPanel-button {
+	height: 21px;
+	margin-right: 1px;
+	margin-top: 0px !important;
+	padding: 1px 3px 0 3px;
 }
 
-#typo3-pagetree-topPanel-filterWrap {
-	padding-top: 3px;
-	height: 24px;
+.typo3-pagetree-topPanel-item {
+	padding: 2px 0 0 0;
+	height: 22px;
 }
 
-#typo3-pagetree-topPanel-filter {
-	height: 16px;
-	width: 98% !important;
-	padding: 1px 0 1px 2px;
+#typo3-pagetree-topPanel-item-newNode {
+	padding: 0;
 }
 
-#typo3-pagetree-topPanel-filterWrap .t3-icon-input-clear {
-	right: 2%;
+.typo3-pagetree-topPanel-item .x-form-item {
+	border: none !important;
+	margin: 3px;
+	width: 100%;
 }
 
-#typo3-pagetree .typo3-pagetree-topPanel-button {
-	margin-right: 1px;
-	margin-top: -2px;
-	padding: 2px 3px;
+.typo3-pagetree-topPanel-item .x-form-field {
+	height: 18px;
+	line-height: 16px;
 }
 
-#typo3-pagetree-topPanel-defaultPanel {
-	padding-left: 6px;
+.typo3-pagetree-topPanel-filter .t3-icon-input-clear {
 }
 
-#typo3-pagetree-topPanel-filterWrap .x-form-field-trigger-wrap {
-	width: 100% !important;
-	margin: 0 20px 0 0;
+.typo3-pagetree-topPanel-item .x-form-trigger-wrap {
+	margin: 0;
 }
 
 .typo3-pagetree-topPanel-item .x-form-trigger {
-	margin: 4px 0 0;
+	background-image: none;
+	border-bottom: none !important;
 	display: none;
+	margin-right: 0;
+	margin-top: 1px;
 }
 
-.typo3-pagetree-topPanel-item .x-form-field-trigger-wrap:hover .x-form-trigger {
+.typo3-pagetree-topPanel-item .x-form-trigger-wrap-focus:hover .x-form-trigger {
 	display: block;
 }
 
@@ -154,25 +136,31 @@ Pagetree
  * @section Indicator Bar
  *
  */
-.typo3-pagetree-indicatorBar-item p {
-	padding: 5px 10px;
+#typo3-pagetree-indicatorBar {
+	width: 100%;
+}
+
+#typo3-pagetree-indicatorBar-filter,
+#typo3-pagetree-indicatorBar-temporaryMountPoint {
+	padding: 10px 20px;	
 }
 
 #typo3-pagetree-indicatorBar-filter p,
 #typo3-pagetree-indicatorBar-temporaryMountPoint p {
-	padding: 10px 35px;
+	padding-left: 5px;
+	width: 100%;
 }
 
 .typo3-pagetree-indicatorBar-item .typo3-pagetree-indicatorBar-item-leftIcon {
-	left: 10px;
-	top: 9px;
-	position: absolute;
+	position: relative;
+	left: -10px;
+	top: 0px;
 }
 
 .typo3-pagetree-indicatorBar-item .typo3-pagetree-indicatorBar-item-rightIcon {
-	right: 10px;
-	top: 5px;
-	position: absolute;
+	position: relative;
+	right: -10px;
+	vertical-align: top;
 }
 
 #typo3-pagetree-indicatorBar-indicatorTitle span {
diff --git a/typo3/sysext/t3skin/stylesheets/visual/element_pagetree.css b/typo3/sysext/t3skin/stylesheets/visual/element_pagetree.css
index 67dd10fc473fcaff1b3a890a46bde53e7a8828ab..2337e495ae04da2a9b9fc2dc3391f5834dc192d1 100644
--- a/typo3/sysext/t3skin/stylesheets/visual/element_pagetree.css
+++ b/typo3/sysext/t3skin/stylesheets/visual/element_pagetree.css
@@ -3,28 +3,29 @@ Pagetree
 
 - - - - - - - - - - - - - - - - - - - - - */
 #typo3-pagetree-treeContainer,
-#typo3-pagetree-topPanelItems {
+.typo3-pageTree-topPanelItems {
 	background-color: #ebebeb;
 }
 
-#typo3-pagetree .x-tree .x-panel-body {
+.typo3-pagetree .x-panel-body {
 	background: none;
 }
 
-#typo3-pagetree .x-tree-node .x-tree-node-el {
+.typo3-pagetree .x-grid-row .x-grid-cell {
 	border-top: 1px solid transparent;
 	border-bottom: 1px solid transparent;
 	border-left: 1px solid transparent;
+	font-size: 12px;
 }
 
-#typo3-pagetree .x-tree-node .x-tree-node-over,
-#typo3-pagetree .x-tree-node .x-tree-selected {
+.typo3-pagetree .x-grid-row.x-grid-row-over .x-grid-cell,
+.typo3-pagetree .x-grid-row.x-grid-row-selected .x-grid-cell {
 	border-bottom: 1px solid #d7d7d7;
 	border-top: 1px solid #d7d7d7;
 	border-left: 1px solid #d7d7d7;
 }
 
-#typo3-pagetree .x-tree-node .x-tree-node-over {
+.typo3-pagetree .x-grid-row.x-grid-row-over .x-grid-cell {
 	background-image: url("");
 	/* star-hack targets IE7 */
 	*background-image: url("../../images/shadows/navigation-container.png");
@@ -39,20 +40,34 @@ Pagetree
 	border-bottom-right-radius: 3px;
 }
 
-#typo3-pagetree .x-tree-node .x-tree-selected {
+.typo3-pagetree .x-grid-row-selected .x-grid-cell {
 	background-color: #f8f8f8;
 }
 
-#typo3-pagetree .x-tree-node .x-tree-selected.x-tree-node-over {
+.typo3-pagetree .x-grid-row-selected.x-grid-row-over .x-grid-cell {
 	background-image: none;
 	-webkit-border-radius: 0;
 	-moz-border-radius: 0;
 	border-radius: 0;
 }
 
-#typo3-pagetree .x-panel-tbar {
-	background-color: #585858;
-	border: none;
+.typo3-pagetree .x-tree-elbow-plus,
+.typo3-pagetree .x-tree-elbow-end-plus {
+	background-image:url('../../images/icons/apps/pagetree-collapse.png');
+}
+
+.typo3-pagetree .x-tree-elbow-minus,
+.typo3-pagetree .x-tree-elbow-end-minus,
+.typo3-pagetree .x-grid-tree-node-expanded .x-tree-elbow-plus,
+.typo3-pagetree .x-grid-tree-node-expanded .x-tree-elbow-end-plus {
+	background-image:url('../../images/icons/apps/pagetree-expand.png');
+}
+
+.typo3-pagetree .x-form-field {
+	color: black;
+	font-family: Verdana,Arial,Helvetica,sans-serif;
+	font-size: 11px;
+	line-height: 14px;
 }
 
 /**
@@ -62,6 +77,8 @@ Pagetree
  */
 .typo3-pagetree-tree-copy {
 	color: #666;
+	font-size: 11px;
+	text-align: center;
 }
 
 .x-dd-drag-ghost {
@@ -149,7 +166,13 @@ Pagetree
  * @section Top Panel
  *
  */
-#typo3-pagetree .x-toolbar {
+#typo3-pagetree-topPanel .x-docked-top,
+#typo3-pagetree-topPanel .x-docked-top .x-box-inner {
+	background-color: #585858;
+	border: none;
+}
+
+#typo3-pagetree .x-panel-body .x-toolbar {
 	background: none;
 	border: none;
 }
@@ -159,6 +182,10 @@ Pagetree
 	border: none;
 }
 
+#typo3-pagetree-topPanel-body {
+	background-color: #dadada;
+}
+
 #typo3-pagetree-topPanel .x-btn-click,
 #typo3-pagetree-topPanel .x-btn-over {
 	background-image: url("");
@@ -193,19 +220,6 @@ Pagetree
 
 #typo3-pagetree .typo3-pagetree-topPanel-item {
 	background-color: #dadada;
-	line-height: normal;
-}
-
-#typo3-pagetree-topPanel-filter {
-	border: 1px solid #aeaeae;
-	-moz-box-shadow: inset 0 1px 1px #aeaeae;
-	-ms-box-shadow: inset 0 1px 1px #aeaeae;
-	-webkit-box-shadow: inset 0 1px 1px #aeaeae;
-	box-shadow: inset 0 2px 1px #aeaeae;
-}
-
-.typo3-pagetree-topPanel-filter-defaultText {
-	color: gray;
 }
 
 #typo3-pagetree .typo3-pagetree-topPanel-button {
@@ -219,7 +233,10 @@ Pagetree
 
 #typo3-pagetree-topPanel-defaultPanel {
 	color: #a2aab8;
-	line-height: 18px;
+}
+
+.typo3-pagetree-topPanel-filter-defaultText {
+	color: gray;
 }
 
 #typo3-pagetree-topPanel-item-newNode .x-btn-over {
@@ -271,7 +288,8 @@ Pagetree
  *
  */
 
-#typo3-pagetree-indicatorBar-temporaryMountPoint p {
+#typo3-pagetree-indicatorBar-temporaryMountPoint p,
+#typo3-pagetree-indicatorBar-filter p {
 	line-height: 13px;
 }
 
@@ -296,8 +314,7 @@ Pagetree
  * @section Deletion Drop Zone
  *
  */
-
-#typo3-pagetree-deletionDropZone {
+#typo3-pagetree-deletionDropZone .x-panel-body {
 	color: #FFF;
 	background-color: #585858;
 	-moz-box-shadow: inset 0 2px 5px #414141;