diff --git a/indra/newview/CMakeLists.txt b/indra/newview/CMakeLists.txt index 46a37767c2222c2e96f05b0e67c42753c6857c5a..a4adcd3f37959ff5392dd78b022407181072363b 100644 --- a/indra/newview/CMakeLists.txt +++ b/indra/newview/CMakeLists.txt @@ -122,6 +122,7 @@ set(viewer_SOURCE_FILES alpanelaopulldown.cpp alpanelquicksettings.cpp alpanelquicksettingspulldown.cpp + altoolalign.cpp alunzip.cpp alviewermenu.cpp groupchatlistener.cpp @@ -776,6 +777,7 @@ set(viewer_HEADER_FILES alpanelaopulldown.h alpanelquicksettings.h alpanelquicksettingspulldown.h + altoolalign.h alunzip.h alviewermenu.h groupchatlistener.h diff --git a/indra/newview/altoolalign.cpp b/indra/newview/altoolalign.cpp new file mode 100644 index 0000000000000000000000000000000000000000..d62221d372049dc27270355d20944dcd462fc698 --- /dev/null +++ b/indra/newview/altoolalign.cpp @@ -0,0 +1,568 @@ +/** + * @file qtoolalign.cpp + * @brief A tool to align objects + */ + +#include "llviewerprecompiledheaders.h" + +// File includes +#include "altoolalign.h" + +// Library includes +#include "llfloaterreg.h" +#include "llbbox.h" +#include "v3math.h" + +// Viewer includes +#include "llagentcamera.h" +#include "llbox.h" +#include "llcylinder.h" +#include "llfloatertools.h" +#include "llselectmgr.h" +#include "llviewercamera.h" +#include "llviewercontrol.h" +#include "llviewerobject.h" +#include "llviewerwindow.h" + +const F32 MANIPULATOR_SIZE = 5.0f; +const F32 MANIPULATOR_SELECT_SIZE = 20.0f; +const F32 FUDGE = 0.001f; // because of SL precision/rounding + +ALToolAlign::ALToolAlign() + : LLTool(std::string("Align")) + , mManipulatorSize(MANIPULATOR_SIZE) + , mHighlightedAxis(-1) + , mHighlightedDirection(0) + , mForce(false) + +{ +} + +BOOL ALToolAlign::handleMouseDown(S32 x, S32 y, MASK mask) +{ + if (mHighlightedAxis != -1) + { + align(); + } + else + { + gViewerWindow->pickAsync(x, y, mask, pickCallback); + } + + return TRUE; +} + + +void ALToolAlign::pickCallback(const LLPickInfo& pick_info) +{ + LLViewerObject* object = pick_info.getObject(); + if (object) + { + if (object->isAvatar()) + { + return; + } + + if (pick_info.mKeyMask & MASK_SHIFT) + { + // If object not selected, select it + if (!object->isSelected()) + { + LLSelectMgr::getInstance()->selectObjectAndFamily(object); + } + else + { + LLSelectMgr::getInstance()->deselectObjectAndFamily(object); + } + } + else + { + LLSelectMgr::getInstance()->deselectAll(); + LLSelectMgr::getInstance()->selectObjectAndFamily(object); + } + + } + else + { + if (pick_info.mKeyMask != MASK_SHIFT) + { + LLSelectMgr::getInstance()->deselectAll(); + } + } + + LLSelectMgr::getInstance()->promoteSelectionToRoot(); +} + +void ALToolAlign::handleSelect() +{ + // no parts, please + + LL_DEBUGS("ALIGNTOOL") << "in select" << LL_ENDL; + LLSelectMgr::getInstance()->promoteSelectionToRoot(); +} + +void ALToolAlign::handleDeselect() +{ +} + +BOOL ALToolAlign::findSelectedManipulator(S32 x, S32 y) +{ + mHighlightedAxis = -1; + mHighlightedDirection = 0; + + LLViewerCamera* camera = LLViewerCamera::getInstance(); + + LLMatrix4 transform; + if (LLSelectMgr::getInstance()->getSelection()->getSelectType() == SELECT_TYPE_HUD) + { + LLVector4 translation(mBBox.getCenterAgent()); + transform.initRotTrans(mBBox.getRotation(), translation); + LLMatrix4 cfr(OGL_TO_CFR_ROTATION); + transform *= cfr; + LLMatrix4 window_scale; + F32 zoom_level = 2.f * gAgentCamera.mHUDCurZoom; + window_scale.initAll(LLVector3(zoom_level / camera->getAspect(), zoom_level, 0.f), + LLQuaternion::DEFAULT, + LLVector3::zero); + transform *= window_scale; + } + else + { + transform.initAll(LLVector3(1.f, 1.f, 1.f), mBBox.getRotation(), mBBox.getCenterAgent()); + + LLMatrix4 projection_matrix = camera->getProjection(); + LLMatrix4 model_matrix = camera->getModelview(); + + transform *= model_matrix; + transform *= projection_matrix; + } + + + LLRect world_view_rect = gViewerWindow->getWorldViewRectScaled(); + F32 half_width = (F32)world_view_rect.getWidth() / 2.f; + F32 half_height = (F32)world_view_rect.getHeight() / 2.f; + LLVector2 manip2d; + LLVector2 mousePos((F32)x - half_width, (F32)y - half_height); + LLVector2 delta; + + LLVector3 bbox_scale = mBBox.getMaxLocal() - mBBox.getMinLocal(); + + for (S32 axis = VX; axis <= VZ; axis++) + { + for (F32 direction = -1.0f; direction <= 1.0f; direction += 2.0f) + { + LLVector3 axis_vector = LLVector3(0.f, 0.f, 0.f); + axis_vector.mV[axis] = direction * bbox_scale.mV[axis] / 2.0f; + + LLVector4 manipulator_center = LLVector4(axis_vector); + + LLVector4 screen_center = manipulator_center * transform; + screen_center /= screen_center.mV[VW]; + + manip2d.setVec(screen_center.mV[VX] * half_width, screen_center.mV[VY] * half_height); + + delta = manip2d - mousePos; + + if (delta.magVecSquared() < MANIPULATOR_SELECT_SIZE * MANIPULATOR_SELECT_SIZE) + { + mHighlightedAxis = axis; + mHighlightedDirection = direction; + return TRUE; + } + + } + } + + return FALSE; +} + +BOOL ALToolAlign::handleHover(S32 x, S32 y, MASK mask) +{ + mForce = (mask & MASK_SHIFT) ? false : true; + + gViewerWindow->setCursor(UI_CURSOR_ARROW); + return findSelectedManipulator(x, y); +} + +void setup_transforms_bbox(LLBBox bbox) +{ + // translate to center + LLVector3 center = bbox.getCenterAgent(); + gGL.translatef(center.mV[VX], center.mV[VY], center.mV[VZ]); + + // rotate + LLQuaternion rotation = bbox.getRotation(); + F32 angle_radians, x, y, z; + rotation.getAngleAxis(&angle_radians, &x, &y, &z); + gGL.rotatef(angle_radians * RAD_TO_DEG, x, y, z); + + // scale + LLVector3 scale = bbox.getMaxLocal() - bbox.getMinLocal(); + gGL.scalef(scale.mV[VX], scale.mV[VY], scale.mV[VZ]); +} + +void render_bbox(LLBBox bbox) +{ + gGL.matrixMode(LLRender::MM_MODELVIEW); + gGL.pushMatrix(); + + setup_transforms_bbox(bbox); + + gGL.flush(); + gBox.render(); + + gGL.popMatrix(); +} + +void render_cone_bbox(LLBBox bbox) +{ + gGL.matrixMode(LLRender::MM_MODELVIEW); + gGL.pushMatrix(); + + setup_transforms_bbox(bbox); + + gGL.flush(); + gCone.render(); + + gGL.popMatrix(); +} + +// the selection bbox isn't axis aligned, so we must construct one +// should this be cached in the selection manager? yes. +LLBBox get_selection_axis_aligned_bbox() +{ + LLBBox selection_bbox = LLSelectMgr::getInstance()->getBBoxOfSelection(); + LLVector3 position = selection_bbox.getPositionAgent(); + + LLBBox axis_aligned_bbox = LLBBox(position, LLQuaternion(), LLVector3(), LLVector3()); + axis_aligned_bbox.addPointLocal(LLVector3()); + + // cycle over the nodes in selection + for (LLObjectSelection::iterator selection_iter = LLSelectMgr::getInstance()->getSelection()->begin(); + selection_iter != LLSelectMgr::getInstance()->getSelection()->end(); + ++selection_iter) + { + LLSelectNode *select_node = *selection_iter; + if (select_node) + { + LLViewerObject* object = select_node->getObject(); + if (object) + { + axis_aligned_bbox.addBBoxAgent(object->getBoundingBoxAgent()); + } + } + } + + + return axis_aligned_bbox; +} + +void ALToolAlign::computeManipulatorSize() +{ + LLViewerCamera* camera = LLViewerCamera::getInstance(); + + if (LLSelectMgr::getInstance()->getSelection()->getSelectType() == SELECT_TYPE_HUD) + { + mManipulatorSize = MANIPULATOR_SIZE / (camera->getViewHeightInPixels() * + gAgentCamera.mHUDCurZoom); + } + else + { + F32 distance = dist_vec(gAgentCamera.getCameraPositionAgent(), mBBox.getCenterAgent()); + + if (distance > FUDGE) + { + // range != zero + F32 fraction_of_fov = MANIPULATOR_SIZE / camera->getViewHeightInPixels(); + F32 apparent_angle = fraction_of_fov * camera->getView(); // radians + mManipulatorSize = MANIPULATOR_SIZE * distance * tan(apparent_angle); + } + else + { + // range == zero + mManipulatorSize = MANIPULATOR_SIZE; + } + } + + //Assume that UI scale factor is equivalent for X and Y axi + F32 ui_scale_factor = LLUI::getScaleFactor().mV[VX]; + mManipulatorSize *= ui_scale_factor; +} + +LLColor4 manipulator_color[3] = { LLColor4(0.7f, 0.0f, 0.0f, 0.5f), + LLColor4(0.0f, 0.7f, 0.0f, 0.5f), + LLColor4(0.0f, 0.0f, 0.7f, 0.5f) }; + + +void ALToolAlign::renderManipulators() +{ + computeManipulatorSize(); + LLVector3 bbox_center = mBBox.getCenterAgent(); + LLVector3 bbox_scale = mBBox.getMaxLocal() - mBBox.getMinLocal(); + + for (S32 axis = VX; axis <= VZ; axis++) + { + for (F32 direction = -1.0f; direction <= 1.0f; direction += 2.0f) + { + F32 size = mManipulatorSize; + LLColor4 color = manipulator_color[axis]; + + if ((axis == mHighlightedAxis) && (direction == mHighlightedDirection)) + { + size *= 1.25f; + color *= 1.5f; + } + + S32 arrows = 1; + if (mForce) + { + arrows = 2; + } + + for (S32 i = 0; i < arrows; i++) + { + LLVector3 axis_vector = LLVector3(0.f, 0.f, 0.f); + axis_vector.mV[axis] = direction * (bbox_scale.mV[axis] / 2.0f + i * (size / 3.0f)); + + LLVector3 manipulator_center = bbox_center + axis_vector; + + LLQuaternion manipulator_rotation; + manipulator_rotation.shortestArc(LLVector3(0.f, 0.f, 1.f), -1.0f * axis_vector); + + LLBBox manipulator_bbox = LLBBox(manipulator_center, manipulator_rotation, + LLVector3(), LLVector3()); + + manipulator_bbox.addPointLocal(LLVector3(-1.f, -1.f, -0.75f) * size * 0.5f); + manipulator_bbox.addPointLocal(LLVector3(1.f, 1.f, 0.75f) * size * 0.5f); + + gGL.color4fv(color.mV); + gGL.diffuseColor4fv(color.mV); + + render_cone_bbox(manipulator_bbox); + } + } + } +} + +BOOL ALToolAlign::canAffectSelection() +{ + // An selection is scalable if you are allowed to move the objects + // and it does not have any sitting agents. In case of editing linked parts, + // the object itself has to be modifiable. + static LLCachedControl<bool> edit_linked_parts(gSavedSettings, "EditLinkedParts"); + BOOL can_scale = LLSelectMgr::getInstance()->getSelection()->getObjectCount() != 0; + if (can_scale) + { + struct f : public LLSelectedObjectFunctor + { + virtual bool apply(LLViewerObject* objectp) + { + return (!edit_linked_parts || objectp->permModify()) && objectp->permMove() && !objectp->isSeat(); + } + } func; + can_scale = LLSelectMgr::getInstance()->getSelection()->applyToObjects(&func); + } + return can_scale; +} + +void ALToolAlign::render() +{ + if(canAffectSelection()) + { + mBBox = get_selection_axis_aligned_bbox(); + + // Draw bounding box + LLGLSUIDefault gls_ui; + LLGLEnable gl_blend(GL_BLEND); + LLGLEnable gls_alpha_test(GL_ALPHA_TEST); + LLGLDepthTest gls_depth(GL_FALSE); + gGL.getTexUnit(0)->unbind(LLTexUnit::TT_TEXTURE); + + // render box + LLColor4 default_normal_color( 0.7f, 0.7f, 0.7f, 0.1f ); + gGL.color4fv( default_normal_color.mV ); + + render_bbox(mBBox); + renderManipulators(); + } +} + +// only works for our specialized (AABB, position centered) bboxes +BOOL bbox_overlap(LLBBox bbox1, LLBBox bbox2) +{ + LLVector3 delta = bbox1.getCenterAgent() - bbox2.getCenterAgent(); + + LLVector3 half_extent = (bbox1.getExtentLocal() + bbox2.getExtentLocal()) / 2.0f; + + return ((fabs(delta.mV[VX]) < half_extent.mV[VX] - FUDGE) && + (fabs(delta.mV[VY]) < half_extent.mV[VY] - FUDGE) && + (fabs(delta.mV[VZ]) < half_extent.mV[VZ] - FUDGE)); +} + +// used to sort bboxes before packing +class BBoxCompare +{ +public: + BBoxCompare(S32 axis, F32 direction, std::map<LLPointer<LLViewerObject>, LLBBox >& bboxes) : + mAxis(axis), mDirection(direction), mBBoxes(bboxes) {} + + BOOL operator() (LLViewerObject* object1, LLViewerObject* object2) + { + LLVector3 corner1 = mBBoxes[object1].getCenterAgent() - + mDirection * mBBoxes[object1].getExtentLocal() / 2.0f; + + LLVector3 corner2 = mBBoxes[object2].getCenterAgent() - + mDirection * mBBoxes[object2].getExtentLocal() / 2.0f; + + return mDirection * corner1.mV[mAxis] < mDirection * corner2.mV[mAxis]; + } + + S32 mAxis; + F32 mDirection; + std::map<LLPointer<LLViewerObject>, LLBBox >& mBBoxes; +}; + +void ALToolAlign::align() +{ + // no linkset parts, please + LLSelectMgr::getInstance()->promoteSelectionToRoot(); + + std::vector<LLPointer<LLViewerObject> > objects; + std::map<LLPointer<LLViewerObject>, LLBBox > original_bboxes; + + // cycle over the nodes in selection and collect them into an array + for (LLObjectSelection::root_iterator selection_iter = LLSelectMgr::getInstance()->getSelection()->root_begin(); + selection_iter != LLSelectMgr::getInstance()->getSelection()->root_end(); + ++selection_iter) + { + LLSelectNode *select_node = *selection_iter; + if (select_node) + { + LLViewerObject* object = select_node->getObject(); + if (object) + { + LLVector3 position = object->getPositionAgent(); + + LLBBox bbox = LLBBox(position, LLQuaternion(), LLVector3(), LLVector3()); + bbox.addPointLocal(LLVector3()); + + // add the parent's bbox + bbox.addBBoxAgent(object->getBoundingBoxAgent()); + LLViewerObject::const_child_list_t& children = object->getChildren(); + + for (LLViewerObject* child : children) + { + // add the child's bbox + bbox.addBBoxAgent(child->getBoundingBoxAgent()); + } + + objects.push_back(object); + original_bboxes[object] = bbox; + } + } + } + + S32 axis = mHighlightedAxis; + F32 direction = mHighlightedDirection; + + // sort them into positional order for proper packing + BBoxCompare compare(axis, direction, original_bboxes); + sort(objects.begin(), objects.end(), compare); + + // storage for their new position after alignment - start with original position first + std::map<LLPointer<LLViewerObject>, LLBBox > new_bboxes = original_bboxes; + + // find new positions + for (size_t i = 0; i < objects.size(); i++) + { + LLBBox target_bbox = mBBox; + LLVector3 target_corner = target_bbox.getCenterAgent() - + direction * target_bbox.getExtentLocal() / 2.0f; + + LLViewerObject* object = objects[i]; + + LLBBox this_bbox = original_bboxes[object]; + LLVector3 this_corner = this_bbox.getCenterAgent() - + direction * this_bbox.getExtentLocal() / 2.0f; + + // for packing, we cycle over several possible positions, taking the smallest that does not overlap + F32 smallest = direction * 9999999.f; // 999999 guarenteed not to be the smallest + for (size_t j = 0; j <= i; j++) + { + // how far must it move? + LLVector3 delta = target_corner - this_corner; + + // new position moves only on one axis, please + LLVector3 delta_one_axis = LLVector3(0.f, 0.f, 0.f); + delta_one_axis.mV[axis] = delta.mV[axis]; + + LLVector3 new_position = this_bbox.getCenterAgent() + delta_one_axis; + + // construct the new bbox + LLBBox new_bbox = LLBBox(new_position, LLQuaternion(), LLVector3(), LLVector3()); + new_bbox.addPointLocal(this_bbox.getExtentLocal() / 2.0f); + new_bbox.addPointLocal(-1.0f * this_bbox.getExtentLocal() / 2.0f); + + // check to see if it overlaps the previously placed objects + BOOL overlap = FALSE; + + //LL_DEBUGS("ToolAlign") << "i=" << i << " j=" << j << LL_ENDL; + + if (!mForce) // well, don't check if in force mode + { + for (size_t k = 0; k < i; k++) + { + LLViewerObject* other_object = objects[k]; + LLBBox other_bbox = new_bboxes[other_object]; + + BOOL overlaps_this = bbox_overlap(other_bbox, new_bbox); + + //if (overlaps_this) + //{ + // LL_WARNS("ToolAlign") << "overlap" << new_bbox.getCenterAgent() << other_bbox.getCenterAgent() << LL_ENDL; + // LL_WARNS("ToolAlign") << "extent" << new_bbox.getExtentLocal() << other_bbox.getExtentLocal() << LL_ENDL; + //} + + overlap = (overlap || overlaps_this); + } + } + + if (!overlap) + { + F32 this_value = (new_bbox.getCenterAgent() - + direction * new_bbox.getExtentLocal() / 2.0f).mV[axis]; + + if (direction * this_value < direction * smallest) + { + smallest = this_value; + // store it + new_bboxes[object] = new_bbox; + } + } + + // update target for next time through the loop + if (j < objects.size()) + { + LLBBox next_bbox = new_bboxes[objects[j]]; + target_corner = next_bbox.getCenterAgent() + + direction * next_bbox.getExtentLocal() / 2.0f; + } + } + } + + // now move them + for (LLViewerObject* object : objects) + { + LLBBox original_bbox = original_bboxes[object]; + LLBBox new_bbox = new_bboxes[object]; + + LLVector3 delta = new_bbox.getCenterAgent() - original_bbox.getCenterAgent(); + + LLVector3 original_position = object->getPositionAgent(); + LLVector3 new_position = original_position + delta; + + object->setPosition(new_position); + } + + LLSelectMgr::getInstance()->sendMultipleUpdate(UPD_POSITION); +} diff --git a/indra/newview/altoolalign.h b/indra/newview/altoolalign.h new file mode 100644 index 0000000000000000000000000000000000000000..1d5fc552d629cae40f743a79140cac2375fb56df --- /dev/null +++ b/indra/newview/altoolalign.h @@ -0,0 +1,46 @@ +/** + * @file qtoolalign.h + * @brief A tool to align objects + */ + +#ifndef AL_ALTOOLALIGN_H +#define AL_ALTOOLALIGN_H + +#include "llsingleton.h" +#include "lltool.h" +#include "llbbox.h" + +class LLViewerObject; +class LLPickInfo; +class LLToolSelectRect; + +class ALToolAlign final +: public LLTool, public LLSingleton<ALToolAlign> +{ + LLSINGLETON(ALToolAlign); + ~ALToolAlign() = default; + +public: + void handleSelect() override; + void handleDeselect() override; + BOOL handleMouseDown(S32 x, S32 y, MASK mask) override; + BOOL handleHover(S32 x, S32 y, MASK mask) override; + void render() override; + BOOL canAffectSelection(); + + static void pickCallback(const LLPickInfo& pick_info); + +private: + void align(); + void computeManipulatorSize(); + void renderManipulators(); + BOOL findSelectedManipulator(S32 x, S32 y); + + LLBBox mBBox; + F32 mManipulatorSize; + S32 mHighlightedAxis; + F32 mHighlightedDirection; + bool mForce; +}; + +#endif // AL_ALTOOLALIGN_H diff --git a/indra/newview/llfloatertools.cpp b/indra/newview/llfloatertools.cpp index 3577649be1d9658aa3f8062725b90e6279e13bc0..36f2aee1cc81d1b90c654b43b489e5167524ffe4 100644 --- a/indra/newview/llfloatertools.cpp +++ b/indra/newview/llfloatertools.cpp @@ -32,6 +32,7 @@ #include "llcoord.h" //#include "llgl.h" +#include "altoolalign.h" #include "llagent.h" #include "llagentcamera.h" #include "llbutton.h" @@ -656,6 +657,7 @@ void LLFloaterTools::updatePopup(LLCoordGL center, MASK mask) tool == LLToolCompScale::getInstance() || tool == LLToolFace::getInstance() || tool == LLToolIndividual::getInstance() || + tool == ALToolAlign::getInstance() || tool == LLToolPipette::getInstance(); mBtnEdit ->setToggleState( edit_visible ); @@ -691,8 +693,11 @@ void LLFloaterTools::updatePopup(LLCoordGL center, MASK mask) { mRadioGroupEdit->setValue("radio select face"); } + else if ( tool == ALToolAlign::getInstance() ) + { + mRadioGroupEdit->setValue("radio align"); + } - if (mComboGridMode) { mComboGridMode->setVisible( edit_visible ); S32 index = mComboGridMode->getCurrentIndex(); @@ -1009,6 +1014,10 @@ void commit_radio_group_edit(LLUICtrl *ctrl) { LLFloaterTools::setEditTool( LLToolFace::getInstance() ); } + else if (selected == "radio align") + { + LLFloaterTools::setEditTool( ALToolAlign::getInstance() ); + } gSavedSettings.setBOOL("ShowParcelOwners", show_owners); } diff --git a/indra/newview/skins/default/xui/en/floater_tools.xml b/indra/newview/skins/default/xui/en/floater_tools.xml index d92edd6d464448843d9601795ebfee6830b3b220..64d3bb2de48448285866ea38475eece5527328c7 100644 --- a/indra/newview/skins/default/xui/en/floater_tools.xml +++ b/indra/newview/skins/default/xui/en/floater_tools.xml @@ -229,7 +229,7 @@ <radio_group follows="left|top" left="5" - top="55" + top="53" height="70" layout="topleft" name="edit_radio_group"> @@ -238,20 +238,25 @@ layout="topleft" name="radio position" /> <radio_item - top_pad="6" + top_pad="5" label="Rotate (Ctrl)" layout="topleft" name="radio rotate" /> <radio_item - top_pad="6" + top_pad="5" label="Stretch (Ctrl+Shift)" layout="topleft" name="radio stretch" /> <radio_item - top_pad="6" + top_pad="5" label="Select Face" layout="topleft" name="radio select face" /> + <radio_item + top_pad="5" + label="Align" + layout="topleft" + name="radio align" /> <radio_group.commit_callback function="BuildTool.commitRadioEdit"/> </radio_group> @@ -263,7 +268,7 @@ label="Edit linked" layout="topleft" name="checkbox edit linked parts" - top_pad="-18"> + top_pad="-8"> <check_box.commit_callback function="BuildTool.selectComponent"/> </check_box> @@ -272,7 +277,7 @@ follows="left|top" height="20" label="Link" - top="108" + top="103" layout="topleft" left="143" name="link_btn" @@ -750,7 +755,7 @@ layout="topleft" left="10" name="selection_empty" - top_pad="0" + top_pad="11" width="100"> Nothing selected. </text> @@ -778,7 +783,7 @@ layout="topleft" left="10" name="remaining_capacity" - top_pad="0" + top_pad="-2" visible="false" width="280"> [CAPACITY_STRING] [secondlife:///app/openfloater/object_weights More info] @@ -816,7 +821,7 @@ layout="topleft" left="6" name="cost_text_border" - top="135" + top="145" width="282"/> <tab_container follows="left|top" @@ -827,9 +832,8 @@ tab_max_width="100" tab_min_width="40" tab_position="top" - tab_height="25" open_tabs_on_drag_and_drop="true" - top="173" + top="180" width="295"> <panel