/* Copyright (c) 2018, Lars Brubaker, John Lewin All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. The views and conclusions contained in the software and documentation are those of the authors and should not be interpreted as representing official policies, either expressed or implied, of the FreeBSD Project. */ using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.IO; using System.Linq; using System.Threading.Tasks; using MatterHackers.Agg; using MatterHackers.Agg.Image; using MatterHackers.Agg.Platform; using MatterHackers.Agg.UI; using MatterHackers.DataConverters3D; using MatterHackers.Localizations; using MatterHackers.MatterControl.CustomWidgets; using MatterHackers.MatterControl.DataStorage; using MatterHackers.MatterControl.DesignTools.Operations; using MatterHackers.MatterControl.Library; using MatterHackers.MatterControl.PrintLibrary; using MatterHackers.MatterControl.SlicerConfiguration; using MatterHackers.VectorMath; namespace MatterHackers.MatterControl.PartPreviewWindow { public enum ViewControls3DButtons { Rotate, Scale, Translate, PartSelect } public enum PartViewMode { Layers2D, Layers3D, Model } public class ViewModeChangedEventArgs : EventArgs { public PartViewMode ViewMode { get; set; } public PartViewMode PreviousMode { get; set; } } public class TransformStateChangedEventArgs : EventArgs { public ViewControls3DButtons TransformMode { get; set; } } public class ViewControls3D : OverflowBar { public event EventHandler ResetView; public event EventHandler TransformStateChanged; private RadioIconButton translateButton; private RadioIconButton rotateButton; private RadioIconButton scaleButton; private RadioIconButton partSelectButton; private View3DWidget view3DWidget; private ISceneContext sceneContext; private PartWorkspace workspace; private ViewControls3DButtons activeTransformState = ViewControls3DButtons.PartSelect; private List<(GuiWidget button, SceneSelectionOperation operation)> operationButtons; private MainViewWidget mainViewWidget = null; private PopupMenuButton bedMenuButton; private ThemeConfig theme; private UndoBuffer undoBuffer; private IconButton undoButton; private IconButton redoButton; internal void NotifyResetView() { this.ResetView.Invoke(this, null); } public bool IsPrinterMode { get; } public ViewControls3DButtons ActiveButton { get => activeTransformState; set { this.activeTransformState = value; switch (this.activeTransformState) { case ViewControls3DButtons.Rotate: if (rotateButton != null) { rotateButton.Checked = true; } break; case ViewControls3DButtons.Translate: if (translateButton != null) { translateButton.Checked = true; } break; case ViewControls3DButtons.Scale: if (scaleButton != null) { scaleButton.Checked = true; } break; case ViewControls3DButtons.PartSelect: if (partSelectButton != null) { partSelectButton.Checked = true; } break; } TransformStateChanged?.Invoke(this, new TransformStateChangedEventArgs() { TransformMode = activeTransformState }); } } internal void SetView3DWidget(View3DWidget view3DWidget) { this.view3DWidget = view3DWidget; var workspaceActions = view3DWidget.WorkspaceActions; bedMenuButton.DynamicPopupContent = () => { var menuTheme = ApplicationController.Instance.MenuTheme; var popupMenu = new PopupMenu(menuTheme); int thumbWidth = 24; popupMenu.CreateSubMenu("Open Recent".Localize(), menuTheme, (subMenu) => { // Select the 25 most recent files and project onto FileSystemItems var recentFiles = new DirectoryInfo(ApplicationDataStorage.Instance.PlatingDirectory).GetFiles("*.mcx").OrderByDescending(f => f.LastWriteTime); foreach (var item in recentFiles.Where(f => f.Length > 500).Select(f => new SceneReplacementFileItem(f.FullName)).Take(10).ToList()) { var imageBuffer = new ImageBuffer(thumbWidth, thumbWidth); var bedHistory = subMenu.CreateMenuItem(item.Name, imageBuffer); bedHistory.Click += (s, e) => { UiThread.RunOnIdle(async () => { await ApplicationController.Instance.Tasks.Execute("Saving changes".Localize() + "...", sceneContext.Printer, sceneContext.SaveChanges); await sceneContext.LoadLibraryContent(item); if (sceneContext.Printer != null) { sceneContext.Printer.ViewState.ViewMode = PartViewMode.Model; } }); }; ApplicationController.Instance.Library.LoadItemThumbnail( (icon) => { imageBuffer.CopyFrom(icon); }, (contentProvider) => { if (contentProvider is MeshContentProvider meshContentProvider) { ApplicationController.Instance.Thumbnails.QueueForGeneration(async () => { // Ask the MeshContentProvider to RayTrace the image var thumbnail = await meshContentProvider.GetThumbnail(item, thumbWidth, thumbWidth); if (thumbnail != null) { //if (thumbnail.Width != thumbWidth //|| thumbnail.Height != thumbHeight) //{ // this.SetUnsizedThumbnail(thumbnail); //} //else //{ // this.SetSizedThumbnail(thumbnail); //} imageBuffer.CopyFrom(thumbnail); popupMenu.Invalidate(); } }); } }, item, ApplicationController.Instance.Library.PlatingHistory, thumbWidth, thumbWidth, menuTheme).ConfigureAwait(false); } }); var actions = new NamedAction[] { new ActionSeparator(), workspaceActions["Cut"], workspaceActions["Copy"], workspaceActions["Paste"], new ActionSeparator(), workspaceActions["Print"], new ActionSeparator(), new NamedAction() { ID = "Export", Title = "Export".Localize(), Icon = AggContext.StaticData.LoadIcon("cube_export.png", 16, 16, menuTheme.InvertIcons), Action = () => { ApplicationController.Instance.ExportLibraryItems( new[] { new InMemoryLibraryItem(sceneContext.Scene)}, centerOnBed: false, printer: view3DWidget.Printer); }, IsEnabled = () => sceneContext.EditableScene || (sceneContext.EditContext.SourceItem is ILibraryAsset libraryAsset && string.Equals(Path.GetExtension(libraryAsset.FileName) ,".gcode" ,StringComparison.OrdinalIgnoreCase)) }, new NamedAction() { ID = "ArrangeAll", Title = "Arrange All Parts".Localize(), Action = async () => { await sceneContext.Scene.AutoArrangeChildren(view3DWidget.BedCenter).ConfigureAwait(false); }, IsEnabled = () => sceneContext.EditableScene }, new ActionSeparator(), new NamedAction() { ID = "ClearBed", Title = "Clear Bed".Localize(), Action = () => { UiThread.RunOnIdle(() => { view3DWidget.ClearPlate(); }); } } }; menuTheme.CreateMenuItems(popupMenu, actions, emptyMenu: false); return popupMenu; }; } public ViewControls3D(PartWorkspace workspace, ThemeConfig theme, UndoBuffer undoBuffer, bool isPrinterType, bool showPrintButton) : base(theme) { this.theme = theme; this.undoBuffer = undoBuffer; this.ActionArea.Click += (s, e) => { view3DWidget.InteractionLayer.Focus(); }; this.IsPrinterMode = isPrinterType; this.sceneContext = workspace.SceneContext; this.workspace = workspace; string iconPath; this.AddChild(CreateAddButton(sceneContext, theme)); this.AddChild(new ToolbarSeparator(theme)); bedMenuButton = new PopupMenuButton(AggContext.StaticData.LoadIcon("bed.png", 16, 16, theme.InvertIcons), theme) { Name = "Bed Options Menu", ToolTipText = "Bed", Enabled = true, Margin = theme.ButtonSpacing, VAnchor = VAnchor.Center, DrawArrow = true }; this.AddChild(bedMenuButton); this.AddChild(new ToolbarSeparator(theme)); this.AddChild(CreateOpenButton(theme)); this.AddChild(CreateSaveButton(theme)); this.AddChild(new ToolbarSeparator(theme)); undoButton = new IconButton(AggContext.StaticData.LoadIcon("Undo_grey_16x.png", 16, 16, theme.InvertIcons), theme) { Name = "3D View Undo", ToolTipText = "Undo".Localize(), Enabled = false, Margin = theme.ButtonSpacing, VAnchor = VAnchor.Center }; undoButton.Click += (sender, e) => { sceneContext.Scene.Undo(); view3DWidget.InteractionLayer.Focus(); }; this.AddChild(undoButton); redoButton = new IconButton(AggContext.StaticData.LoadIcon("Redo_grey_16x.png", 16, 16, theme.InvertIcons), theme) { Name = "3D View Redo", Margin = theme.ButtonSpacing, ToolTipText = "Redo".Localize(), Enabled = false, VAnchor = VAnchor.Center }; redoButton.Click += (sender, e) => { sceneContext.Scene.Redo(); view3DWidget.InteractionLayer.Focus(); }; this.AddChild(redoButton); if (showPrintButton) { var printButton = new TextButton("Print", theme) { Name = "Print Button", BackgroundColor = theme.AccentMimimalOverlay }; printButton.Click += (s, e) => { view3DWidget.PushToPrinterAndPrint(); }; this.AddChild(printButton); } this.AddChild(new ToolbarSeparator(theme)); undoButton.Enabled = undoBuffer.UndoCount > 0; redoButton.Enabled = undoBuffer.RedoCount > 0; var buttonGroupA = new ObservableCollection(); if (UserSettings.Instance.IsTouchScreen) { iconPath = Path.Combine("ViewTransformControls", "rotate.png"); rotateButton = new RadioIconButton(AggContext.StaticData.LoadIcon(iconPath, 32, 32, theme.InvertIcons), theme) { SiblingRadioButtonList = buttonGroupA, ToolTipText = "Rotate (Alt + Left Mouse)".Localize(), Margin = theme.ButtonSpacing }; rotateButton.Click += (s, e) => this.ActiveButton = ViewControls3DButtons.Rotate; buttonGroupA.Add(rotateButton); AddChild(rotateButton); iconPath = Path.Combine("ViewTransformControls", "translate.png"); translateButton = new RadioIconButton(AggContext.StaticData.LoadIcon(iconPath, 32, 32, theme.InvertIcons), theme) { SiblingRadioButtonList = buttonGroupA, ToolTipText = "Move (Shift + Left Mouse)".Localize(), Margin = theme.ButtonSpacing }; translateButton.Click += (s, e) => this.ActiveButton = ViewControls3DButtons.Translate; buttonGroupA.Add(translateButton); AddChild(translateButton); iconPath = Path.Combine("ViewTransformControls", "scale.png"); scaleButton = new RadioIconButton(AggContext.StaticData.LoadIcon(iconPath, 32, 32, theme.InvertIcons), theme) { SiblingRadioButtonList = buttonGroupA, ToolTipText = "Zoom (Ctrl + Left Mouse)".Localize(), Margin = theme.ButtonSpacing }; scaleButton.Click += (s, e) => this.ActiveButton = ViewControls3DButtons.Scale; buttonGroupA.Add(scaleButton); AddChild(scaleButton); rotateButton.Checked = true; // Add vertical separator this.AddChild(new ToolbarSeparator(theme)); iconPath = Path.Combine("ViewTransformControls", "partSelect.png"); partSelectButton = new RadioIconButton(AggContext.StaticData.LoadIcon(iconPath, 32, 32, theme.InvertIcons), theme) { SiblingRadioButtonList = buttonGroupA, ToolTipText = "Select Part".Localize(), Margin = theme.ButtonSpacing }; partSelectButton.Click += (s, e) => this.ActiveButton = ViewControls3DButtons.PartSelect; buttonGroupA.Add(partSelectButton); AddChild(partSelectButton); } operationButtons = new List<(GuiWidget, SceneSelectionOperation)>(); // Add Selected IObject3D -> Operations to toolbar foreach (var namedAction in ApplicationController.Instance.RegisteredSceneOperations) { if (namedAction is SceneSelectionSeparator) { this.AddChild(new ToolbarSeparator(theme)); continue; } GuiWidget button; if (namedAction.Icon != null) { // add the create support before the align if (namedAction.OperationType == typeof(AlignObject3D)) { this.AddChild(CreateWipeTowerButton(theme)); this.AddChild(CreateSupportButton(theme)); } button = new IconButton(namedAction.Icon, theme) { Name = namedAction.Title + " Button", ToolTipText = namedAction.Title, Margin = theme.ButtonSpacing, BackgroundColor = theme.ToolbarButtonBackground, HoverColor = theme.ToolbarButtonHover, MouseDownColor = theme.ToolbarButtonDown, }; } else { button = new TextButton(namedAction.Title, theme) { Name = namedAction.Title + " Button", Margin = theme.ButtonSpacing, BackgroundColor = theme.ToolbarButtonBackground, HoverColor = theme.ToolbarButtonHover, MouseDownColor = theme.ToolbarButtonDown, }; } operationButtons.Add((button, namedAction)); button.Click += (s, e) => { UiThread.RunOnIdle(() => { namedAction.Action.Invoke(sceneContext); var partTab = button.Parents().FirstOrDefault(); var view3D = partTab.Descendants().FirstOrDefault(); view3D.InteractionLayer.Focus(); }); }; this.AddChild(button); } // Register listeners undoBuffer.Changed += UndoBuffer_Changed; sceneContext.Scene.SelectionChanged += Scene_SelectionChanged; // Run on load Scene_SelectionChanged(null, null); } private void UndoBuffer_Changed(object sender, EventArgs e) { undoButton.Enabled = undoBuffer.UndoCount > 0; redoButton.Enabled = undoBuffer.RedoCount > 0; } private IconButton CreateOpenButton(ThemeConfig theme) { var openButton = new IconButton(AggContext.StaticData.LoadIcon("fa-folder-open_16.png", 16, 16, theme.InvertIcons), theme) { Margin = theme.ButtonSpacing, ToolTipText = "Open File".Localize(), Name = "Open File Button" }; openButton.Click += (s, e) => { var extensionsWithoutPeriod = new HashSet(ApplicationSettings.OpenDesignFileParams.Split('|').First().Split(',').Select(t => t.Trim().Trim('.'))); foreach (var extension in ApplicationController.Instance.Library.ContentProviders.Keys) { extensionsWithoutPeriod.Add(extension.ToUpper()); } var extensionsArray = extensionsWithoutPeriod.OrderBy(t => t).ToArray(); string filter = string.Format( "{0}|{1}", string.Join(",", extensionsArray), string.Join("", extensionsArray.Select(t => $"*.{t.ToLower()};").ToArray())); UiThread.RunOnIdle(() => { AggContext.FileDialogs.OpenFileDialog( new OpenFileDialogParams(filter, multiSelect: true), (openParams) => { sceneContext.AddToPlate(openParams.FileNames); }); }, .1); }; return openButton; } private void Scene_SelectionChanged(object sender, EventArgs e) { // Set enabled level based on operation rules foreach(var item in operationButtons) { item.button.Enabled = item.operation.IsEnabled?.Invoke(sceneContext.Scene) ?? false; } } private GuiWidget CreateAddButton(ISceneContext sceneContext, ThemeConfig theme) { var buttonView = new TextIconButton( "", AggContext.StaticData.LoadIcon("cube_add.png", 16, 16, theme.InvertIcons), theme); // Remove right Padding for drop style buttonView.Padding = buttonView.Padding.Clone(right: 0); var popupMenu = new PopupMenu(ApplicationController.Instance.MenuTheme) { HAnchor = HAnchor.Fit, VAnchor = VAnchor.Fit, }; var openColor = theme.ResolveColor(theme.BackgroundColor, theme.SlightShade); PopupMenuButton libraryPopup = null; libraryPopup = new PopupMenuButton(buttonView, theme) { MakeScrollable = false, Name = "Add Content Menu", ToolTipText = "Add Content".Localize(), AlwaysKeepOpen = true, DynamicPopupContent = () => { if (mainViewWidget == null) { mainViewWidget = this.Parents().FirstOrDefault(); } var verticalResizeContainer = new VerticalResizeContainer(theme, GrabBarSide.Right) { BackgroundColor = openColor, MinimumSize = new Vector2(120, 50), Height = libraryPopup.TransformToScreenSpace(libraryPopup.Position).Y, SplitterBarColor = theme.SlightShade, }; double.TryParse(UserSettings.Instance.get(UserSettingsKey.PopupLibraryWidth), out double controlWidth); if (controlWidth == 0) { controlWidth = 400; } verticalResizeContainer.Width = controlWidth; verticalResizeContainer.BoundsChanged += (s2, e2) => { UserSettings.Instance.set(UserSettingsKey.PopupLibraryWidth, verticalResizeContainer.Width.ToString()); }; var systemWindow = this.Parents().FirstOrDefault(); var printLibraryWidget = new PrintLibraryWidget(mainViewWidget, workspace, theme, libraryPopup) { HAnchor = HAnchor.Stretch, VAnchor = VAnchor.Absolute, Height = libraryPopup.TransformToScreenSpace(libraryPopup.Position).Y, Margin = new BorderDouble(left: verticalResizeContainer.SplitterWidth) }; systemWindow.SizeChanged += (s, e) => { printLibraryWidget.Height = libraryPopup.TransformToScreenSpace(libraryPopup.Position).Y; }; verticalResizeContainer.AddChild(printLibraryWidget); systemWindow.MouseDown += systemWindownMouseDown; void systemWindownMouseDown(object s2, MouseEventArgs mouseEvent) { if (verticalResizeContainer.Parent != null) { // MouseUp on our SystemWindow outside of our bounds should call close var resizeContainerMousePosition = verticalResizeContainer.TransformFromScreenSpace(mouseEvent.Position); bool mouseUpOnWidget = resizeContainerMousePosition.X >= 0 && resizeContainerMousePosition.X <= verticalResizeContainer.Width && resizeContainerMousePosition.Y >= 0 && resizeContainerMousePosition.Y <= verticalResizeContainer.Height; if (!mouseUpOnWidget) { libraryPopup.CloseMenu(); systemWindow.MouseDown -= systemWindownMouseDown; } } else { systemWindow.MouseDown -= systemWindownMouseDown; } }; return verticalResizeContainer; }, BackgroundColor = theme.ToolbarButtonBackground, HoverColor = theme.ToolbarButtonHover, MouseDownColor = theme.ToolbarButtonDown, OpenColor = openColor, DrawArrow = true, Margin = theme.ButtonSpacing, PopupBorderColor = Color.Transparent, PopupHAnchor = HAnchor.Fit, PopupVAnchor = VAnchor.Fit }; return libraryPopup; } private GuiWidget CreateWipeTowerButton(ThemeConfig theme) { var iconButton = new IconButton( AggContext.StaticData.LoadIcon("wipe_tower.png", 16, 16, theme.InvertIcons), theme) { ToolTipText = "Create Wipe Tower".Localize(), }; iconButton.Click += (s, e) => { var scene = sceneContext.Scene; var selectedItem = scene.SelectedItem; if (selectedItem != null) { bool allAreWipeTower = false; if (selectedItem is SelectionGroupObject3D) { allAreWipeTower = selectedItem.Children.All(i => i.OutputType == PrintOutputTypes.WipeTower); } else { allAreWipeTower = selectedItem.OutputType == PrintOutputTypes.WipeTower; } scene.UndoBuffer.AddAndDo(new SetOutputType(selectedItem, allAreWipeTower ? PrintOutputTypes.Default : PrintOutputTypes.WipeTower)); } }; sceneContext.Scene.SelectionChanged += (s, e) => { iconButton.Enabled = sceneContext.Scene.SelectedItem != null; }; return iconButton; } private GuiWidget CreateSupportButton(ThemeConfig theme) { PopupMenuButton toggleSupportButton = null; var iconButton = new IconButton( AggContext.StaticData.LoadIcon("support.png", 16, 16, theme.InvertIcons), theme) { ToolTipText = "Toggle Support".Localize(), }; iconButton.Click += (s, e) => { var scene = sceneContext.Scene; var selectedItem = scene.SelectedItem; if (selectedItem != null) { bool allAreSupport = false; if (selectedItem is SelectionGroupObject3D) { allAreSupport = selectedItem.Children.All(i => i.OutputType == PrintOutputTypes.Support); } else { allAreSupport = selectedItem.OutputType == PrintOutputTypes.Support; } scene.UndoBuffer.AddAndDo(new SetOutputType(selectedItem, allAreSupport ? PrintOutputTypes.Default : PrintOutputTypes.Support)); } }; sceneContext.Scene.SelectionChanged += (s, e) => { iconButton.Selectable = sceneContext.Scene.SelectedItem != null; }; // Remove right Padding for drop style iconButton.Padding = iconButton.Padding.Clone(right: 0); var minimumSupportHeight = .05; if (sceneContext.Printer != null) { minimumSupportHeight = sceneContext.Printer.Settings.GetValue(SettingsKey.layer_height) / 2; } toggleSupportButton = new PopupMenuButton(iconButton, theme) { Name = "Support SplitButton", ToolTipText = "Generate Support".Localize(), DynamicPopupContent = () => new GenerateSupportPanel(AppContext.MenuTheme, sceneContext.Scene, minimumSupportHeight), PopupHAnchor = HAnchor.Fit, PopupVAnchor = VAnchor.Fit, MakeScrollable = false, BackgroundColor = theme.ToolbarButtonBackground, HoverColor = theme.ToolbarButtonHover, MouseDownColor = theme.ToolbarButtonDown, DrawArrow = true, Margin = theme.ButtonSpacing, }; iconButton.Selectable = true; return toggleSupportButton; } private GuiWidget CreateSaveButton(ThemeConfig theme) { PopupMenuButton saveButton = null; var iconButton = new IconButton( AggContext.StaticData.LoadIcon("save_grey_16x.png", 16, 16, theme.InvertIcons), theme) { ToolTipText = "Save".Localize(), }; iconButton.Click += (s, e) => { ApplicationController.Instance.Tasks.Execute("Saving".Localize(), sceneContext.Printer, async(progress, cancellationToken) => { saveButton.Enabled = false; try { await sceneContext.SaveChanges(progress, cancellationToken); } catch { } saveButton.Enabled = true; }).ConfigureAwait(false); }; // Remove right Padding for drop style iconButton.Padding = iconButton.Padding.Clone(right: 0); saveButton = new PopupMenuButton(iconButton, theme) { Name = "Save SplitButton", ToolTipText = "Save As".Localize(), DynamicPopupContent = () => { var popupMenu = new PopupMenu(ApplicationController.Instance.MenuTheme); var saveAs = popupMenu.CreateMenuItem("Save As".Localize()); saveAs.Click += (s, e) => UiThread.RunOnIdle(() => { UiThread.RunOnIdle(() => { DialogWindow.Show( new SaveAsPage( async (newName, destinationContainer) => { // Save to the destination provider if (destinationContainer is ILibraryWritableContainer writableContainer) { // Wrap stream with ReadOnlyStream library item and add to container writableContainer.Add(new[] { new InMemoryLibraryItem(sceneContext.Scene) { Name = newName } }); destinationContainer.Dispose(); } })); }); }); return popupMenu; }, BackgroundColor = theme.ToolbarButtonBackground, HoverColor = theme.ToolbarButtonHover, MouseDownColor = theme.ToolbarButtonDown, DrawArrow = true, Margin = theme.ButtonSpacing, }; iconButton.Selectable = true; return saveButton; } public override void OnClosed(EventArgs e) { // Unregister listeners undoBuffer.Changed -= UndoBuffer_Changed; sceneContext.Scene.SelectionChanged -= Scene_SelectionChanged; base.OnClosed(e); } } }