/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2013 Jean-Pierre Charras, jp.charras at wanadoo.fr * Copyright (C) 2013 Wayne Stambaugh * Copyright (C) 1992-2021 KiCad Developers, see AUTHORS.txt for contributors. * * This program 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. * * This program 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. * * You should have received a copy of the GNU General Public License * along with this program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * or you may search the http://www.gnu.org website for the version 2 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include class SCH_EDIT_FRAME; class SCH_TEXT; DIALOG_LABEL_PROPERTIES::DIALOG_LABEL_PROPERTIES( SCH_EDIT_FRAME* aParent, SCH_LABEL_BASE* aLabel ) : DIALOG_LABEL_PROPERTIES_BASE( aParent ), m_activeTextEntry( nullptr ), m_netNameValidator( true ), m_fields( nullptr ), m_textSize( aParent, m_textSizeLabel, m_textSizeCtrl, m_textSizeUnits, false ), m_helpWindow( nullptr ) { m_Parent = aParent; m_currentLabel = aLabel; m_fields = new FIELDS_GRID_TABLE( this, aParent, m_grid, m_currentLabel ); m_width = 100; // Will be later set to a better value m_delayedFocusRow = -1; m_delayedFocusColumn = FDC_VALUE; if( m_currentLabel->Type() == SCH_GLOBAL_LABEL_T || m_currentLabel->Type() == SCH_LABEL_T ) { m_activeTextEntry = m_valueCombo; SetInitialFocus( m_valueCombo ); m_labelSingleLine->Show( false ); m_valueSingleLine->Show( false ); m_valueCombo->SetValidator( m_netNameValidator ); } else if( m_currentLabel->Type() == SCH_HIER_LABEL_T ) { m_activeTextEntry = m_valueSingleLine; SetInitialFocus( m_valueSingleLine ); m_labelCombo->Show( false ); m_valueCombo->Show( false ); m_valueSingleLine->SetValidator( m_netNameValidator ); } else if( m_currentLabel->Type() == SCH_NETCLASS_FLAG_T ) { SetInitialFocus( m_grid ); m_delayedFocusRow = 0; m_labelSingleLine->Show( false ); m_valueSingleLine->Show( false ); m_labelCombo->Show( false ); m_valueCombo->Show( false ); m_syntaxHelp->Show( false ); m_textSizeLabel->SetLabel( _( "Pin length:" ) ); } switch( m_currentLabel->Type() ) { case SCH_GLOBAL_LABEL_T: SetTitle( _( "Global Label Properties" ) ); break; case SCH_HIER_LABEL_T: SetTitle( _( "Hierarchical Label Properties" ) ); break; case SCH_LABEL_T: SetTitle( _( "Label Properties" ) ); break; case SCH_NETCLASS_FLAG_T: SetTitle( _( "Net Class Flag Properties" ) ); break; case SCH_SHEET_PIN_T: SetTitle( _( "Hierarchical Sheet Pin Properties" ) ); break; default: UNIMPLEMENTED_FOR( m_currentLabel->GetClass() ); break; } // Give a bit more room for combobox editors m_grid->SetDefaultRowSize( m_grid->GetDefaultRowSize() + 4 ); m_grid->SetTable( m_fields ); m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this ) ); m_grid->SetSelectionMode( wxGrid::wxGridSelectRows ); // Show/hide columns according to user's preference auto cfg = dynamic_cast( Kiface().KifaceSettings() ); wxASSERT( cfg ); if( cfg ) { m_shownColumns = cfg->m_Appearance.edit_label_visible_columns; m_grid->ShowHideColumns( m_shownColumns ); } // Configure button logos m_bpAdd->SetBitmap( KiBitmap( BITMAPS::small_plus ) ); m_bpDelete->SetBitmap( KiBitmap( BITMAPS::small_trash ) ); m_bpMoveUp->SetBitmap( KiBitmap( BITMAPS::small_up ) ); m_bpMoveDown->SetBitmap( KiBitmap( BITMAPS::small_down ) ); m_separator1->SetIsSeparator(); m_bold->SetIsCheckButton(); m_bold->SetBitmap( KiBitmap( BITMAPS::text_bold ) ); m_italic->SetIsCheckButton(); m_italic->SetBitmap( KiBitmap( BITMAPS::text_italic ) ); m_separator2->SetIsSeparator(); m_spin0->SetIsCheckButton(); m_spin1->SetIsCheckButton(); m_spin2->SetIsCheckButton(); m_spin3->SetIsCheckButton(); m_separator3->SetIsSeparator(); // Show/hide relevant controls if( m_currentLabel->Type() == SCH_GLOBAL_LABEL_T || m_currentLabel->Type() == SCH_HIER_LABEL_T ) { m_dot->Hide(); m_circle->Hide(); m_diamond->Hide(); m_rectangle->Hide(); m_spin0->SetBitmap( KiBitmap( BITMAPS::label_align_left ) ); m_spin1->SetBitmap( KiBitmap( BITMAPS::label_align_right ) ); m_spin2->SetBitmap( KiBitmap( BITMAPS::label_align_bottom ) ); m_spin3->SetBitmap( KiBitmap( BITMAPS::label_align_top ) ); } else if( m_currentLabel->Type() == SCH_NETCLASS_FLAG_T ) { m_input->Hide(); m_output->Hide(); m_bidirectional->Hide(); m_triState->Hide(); m_passive->Hide(); m_spin0->SetBitmap( KiBitmap( BITMAPS::pinorient_down ) ); m_spin1->SetBitmap( KiBitmap( BITMAPS::pinorient_up ) ); m_spin2->SetBitmap( KiBitmap( BITMAPS::pinorient_right ) ); m_spin3->SetBitmap( KiBitmap( BITMAPS::pinorient_left ) ); } else { m_shapeSizer->Show( false ); m_spin0->SetBitmap( KiBitmap( BITMAPS::text_align_left ) ); m_spin1->SetBitmap( KiBitmap( BITMAPS::text_align_right ) ); m_spin2->SetBitmap( KiBitmap( BITMAPS::text_align_bottom ) ); m_spin3->SetBitmap( KiBitmap( BITMAPS::text_align_top ) ); } SetupStandardButtons(); // DIALOG_SHIM needs a unique hash_key because classname is not sufficient because the // various versions have different controls so we want to store sizes for each version. m_hash_key = TO_UTF8( GetTitle() ); m_spin0->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this ); m_spin1->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this ); m_spin2->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this ); m_spin3->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this ); // Now all widgets have the size fixed, call FinishDialogSettings finishDialogSettings(); } DIALOG_LABEL_PROPERTIES::~DIALOG_LABEL_PROPERTIES() { auto cfg = dynamic_cast( Kiface().KifaceSettings() ); wxASSERT( cfg ); if( cfg ) cfg->m_Appearance.edit_sheet_visible_columns = m_grid->GetShownColumns(); // Prevents crash bug in wxGrid's d'tor m_grid->DestroyTable( m_fields ); // Delete the GRID_TRICKS. m_grid->PopEventHandler( true ); if( m_helpWindow ) m_helpWindow->Destroy(); } bool DIALOG_LABEL_PROPERTIES::TransferDataToWindow() { if( !wxDialog::TransferDataToWindow() ) return false; if( m_activeTextEntry ) { // show control characters in a human-readable format m_activeTextEntry->SetValue( UnescapeString( m_currentLabel->GetText() ) ); } if( m_currentLabel->Type() == SCH_GLOBAL_LABEL_T || m_currentLabel->Type() == SCH_LABEL_T ) { // Load the combobox with the existing labels of the same type std::set existingLabels; SCH_SCREENS allScreens( m_Parent->Schematic().Root() ); for( SCH_SCREEN* screen = allScreens.GetFirst(); screen; screen = allScreens.GetNext() ) { for( SCH_ITEM* item : screen->Items().OfType( m_currentLabel->Type() ) ) { const SCH_TEXT* textItem = static_cast( item ); existingLabels.insert( UnescapeString( textItem->GetText() ) ); } } wxArrayString existingLabelArray; for( const wxString& label : existingLabels ) existingLabelArray.push_back( label ); m_valueCombo->Append( existingLabelArray ); } else if( m_currentLabel->Type() == SCH_NETCLASS_FLAG_T ) { // Load the combobox with existing existingNetclassNames NET_SETTINGS& netSettings = m_Parent->Schematic().Prj().GetProjectFile().NetSettings(); wxArrayString existingNetclassNames; existingNetclassNames.push_back( netSettings.m_NetClasses.GetDefault()->GetName() ); for( const std::pair& pair : netSettings.m_NetClasses ) existingNetclassNames.push_back( pair.second->GetName() ); m_valueCombo->Append( existingNetclassNames ); } // Push a copy of each field into m_updateFields for( SCH_FIELD& field : m_currentLabel->GetFields() ) { SCH_FIELD field_copy( field ); // change offset to be symbol-relative field_copy.Offset( -m_currentLabel->GetPosition() ); m_fields->push_back( field_copy ); } // notify the grid wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, m_fields->size() ); m_grid->ProcessTableMessage( msg ); AdjustGridColumns( m_grid->GetRect().GetWidth() ); if( m_shapeSizer->AreAnyItemsShown() ) { switch( m_currentLabel->GetShape() ) { case LABEL_FLAG_SHAPE::L_INPUT: m_input->SetValue( true ); break; case LABEL_FLAG_SHAPE::L_OUTPUT: m_output->SetValue( true ); break; case LABEL_FLAG_SHAPE::L_BIDI: m_bidirectional->SetValue( true ); break; case LABEL_FLAG_SHAPE::L_TRISTATE: m_triState->SetValue( true ); break; case LABEL_FLAG_SHAPE::L_UNSPECIFIED: m_passive->SetValue( true ); break; case LABEL_FLAG_SHAPE::F_DOT: m_dot->SetValue( true ); break; case LABEL_FLAG_SHAPE::F_ROUND: m_circle->SetValue( true ); break; case LABEL_FLAG_SHAPE::F_DIAMOND: m_diamond->SetValue( true ); break; case LABEL_FLAG_SHAPE::F_RECTANGLE: m_rectangle->SetValue( true ); break; } } m_fontCtrl->SetFontSelection( m_currentLabel->GetFont() ); if( m_currentLabel->Type() == SCH_NETCLASS_FLAG_T ) m_textSize.SetValue( static_cast( m_currentLabel )->GetPinLength() ); else m_textSize.SetValue( m_currentLabel->GetTextWidth() ); m_bold->Check( m_currentLabel->IsBold() ); m_italic->Check( m_currentLabel->IsItalic() ); switch( m_currentLabel->GetLabelSpinStyle() ) { case LABEL_SPIN_STYLE::RIGHT: m_spin0->Check( true ); break; case LABEL_SPIN_STYLE::LEFT: m_spin1->Check( true ); break; case LABEL_SPIN_STYLE::UP: m_spin2->Check( true ); break; case LABEL_SPIN_STYLE::BOTTOM: m_spin3->Check( true ); break; } return true; } /*! * wxEVT_COMMAND_ENTER event handler for single-line control */ void DIALOG_LABEL_PROPERTIES::OnEnterKey( wxCommandEvent& aEvent ) { wxPostEvent( this, wxCommandEvent( wxEVT_COMMAND_BUTTON_CLICKED, wxID_OK ) ); } static bool positioningChanged( const SCH_FIELD& a, const SCH_FIELD& b ) { if( a.GetPosition() != b.GetPosition() ) return true; if( a.GetHorizJustify() != b.GetHorizJustify() ) return true; if( a.GetVertJustify() != b.GetVertJustify() ) return true; if( a.GetTextAngle() != b.GetTextAngle() ) return true; return false; } static bool positioningChanged( FIELDS_GRID_TABLE* a, std::vector& b ) { for( size_t i = 0; i < a->size() && i < b.size(); ++i ) { if( positioningChanged( a->at( i ), b.at( i ) ) ) return true; } return false; } bool DIALOG_LABEL_PROPERTIES::TransferDataFromWindow() { if( !m_grid->CommitPendingChanges() ) return false; if( !wxDialog::TransferDataFromWindow() ) return false; // Don't allow text to disappear; it can be difficult to correct if you can't select it if( !m_textSize.Validate( 0.01, 1000.0, EDA_UNITS::MILLIMETRES ) ) return false; wxString text; /* save old text in undo list if not already in edit */ if( m_currentLabel->GetEditFlags() == 0 ) { m_Parent->SaveCopyInUndoList( m_Parent->GetScreen(), m_currentLabel, UNDO_REDO::CHANGED, false ); } m_Parent->GetCanvas()->Refresh(); if( m_activeTextEntry ) { // labels need escaping text = EscapeString( m_activeTextEntry->GetValue(), CTX_NETNAME ); #ifdef __WXMAC__ // On macOS CTRL+Enter produces '\r' instead of '\n' regardless of EOL setting text.Replace( "\r", "\n" ); #endif if( text.IsEmpty() && !m_currentLabel->IsNew() ) { DisplayError( this, _( "Label can not be empty." ) ); return false; } m_currentLabel->SetText( text ); } bool doAutoplace = false; // change all field positions from relative to absolute for( SCH_FIELD& field : *m_fields ) { field.Offset( m_currentLabel->GetPosition() ); if( field.GetCanonicalName() == wxT( "Netclass" ) ) field.SetLayer( LAYER_NETCLASS_REFS ); else if( field.GetCanonicalName() == wxT( "Intersheetrefs" ) ) field.SetLayer( LAYER_INTERSHEET_REFS ); else field.SetLayer( LAYER_FIELDS ); } if( positioningChanged( m_fields, m_currentLabel->GetFields() ) ) m_currentLabel->ClearFieldsAutoplaced(); else doAutoplace = true; m_currentLabel->SetFields( *m_fields ); if( m_shapeSizer->AreAnyItemsShown() ) { if( m_input->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_INPUT ); else if( m_output->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_OUTPUT ); else if( m_bidirectional->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_BIDI ); else if( m_triState->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_TRISTATE ); else if( m_passive->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_UNSPECIFIED ); else if( m_dot->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_DOT ); else if( m_circle->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_ROUND ); else if( m_diamond->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_DIAMOND ); else if( m_rectangle->GetValue() ) m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_RECTANGLE ); } if( m_fontCtrl->HaveFontSelection() ) { m_currentLabel->SetFont( m_fontCtrl->GetFontSelection( m_bold->IsChecked(), m_italic->IsChecked() ) ); } if( m_currentLabel->Type() == SCH_NETCLASS_FLAG_T ) static_cast( m_currentLabel )->SetPinLength( m_textSize.GetValue() ); else if( m_currentLabel->GetTextWidth() != m_textSize.GetValue() ) m_currentLabel->SetTextSize( wxSize( m_textSize.GetValue(), m_textSize.GetValue() ) ); if( m_bold->IsChecked() != m_currentLabel->IsBold() ) { if( m_bold->IsChecked() ) { m_currentLabel->SetBold( true ); m_currentLabel->SetTextThickness( GetPenSizeForBold( m_currentLabel->GetTextWidth() ) ); } else { m_currentLabel->SetBold( false ); m_currentLabel->SetTextThickness( 0 ); // Use default pen width } } m_currentLabel->SetItalic( m_italic->IsChecked() ); LABEL_SPIN_STYLE selectedSpinStyle= LABEL_SPIN_STYLE::LEFT; if( m_spin0->IsChecked() ) selectedSpinStyle = LABEL_SPIN_STYLE::RIGHT; else if( m_spin1->IsChecked() ) selectedSpinStyle = LABEL_SPIN_STYLE::LEFT; else if( m_spin2->IsChecked() ) selectedSpinStyle = LABEL_SPIN_STYLE::UP; else if( m_spin3->IsChecked() ) selectedSpinStyle = LABEL_SPIN_STYLE::BOTTOM; if( m_currentLabel->GetLabelSpinStyle() != selectedSpinStyle ) m_currentLabel->SetLabelSpinStyle( selectedSpinStyle ); if( doAutoplace ) m_currentLabel->AutoAutoplaceFields( m_Parent->GetScreen() ); m_Parent->UpdateItem( m_currentLabel ); m_Parent->GetCanvas()->Refresh(); m_Parent->OnModify(); return true; } void DIALOG_LABEL_PROPERTIES::onSpinButton( wxCommandEvent& aEvent ) { for( BITMAP_BUTTON* btn : { m_spin0, m_spin1, m_spin2, m_spin3 } ) { if( btn->IsChecked() && btn != aEvent.GetEventObject() ) btn->Check( false ); } } void DIALOG_LABEL_PROPERTIES::OnFormattingHelp( wxHyperlinkEvent& aEvent ) { m_helpWindow = SCH_TEXT::ShowSyntaxHelp( this ); } void DIALOG_LABEL_PROPERTIES::OnAddField( wxCommandEvent& event ) { if( !m_grid->CommitPendingChanges() ) return; int fieldID = m_fields->size(); wxString fieldName; if( (int) fieldID == m_currentLabel->GetMandatoryFieldCount() || m_fields->at( m_fields->size()-1 ).GetCanonicalName() == wxT( "Netclass" ) ) { fieldName = wxT( "Netclass" ); } else { fieldName = SCH_LABEL_BASE::GetDefaultFieldName( fieldName, true ); } SCH_FIELD newField( wxPoint( 0, 0 ), fieldID, m_currentLabel, fieldName ); if( m_fields->size() > 0 ) { newField.SetVisible( m_fields->at( m_fields->size() - 1 ).IsVisible() ); newField.SetTextAngle( m_fields->at( m_fields->size() - 1 ).GetTextAngle() ); newField.SetItalic( m_fields->at( m_fields->size() - 1 ).IsItalic() ); newField.SetBold( m_fields->at( m_fields->size() - 1 ).IsBold() ); } else { newField.SetVisible( true ); newField.SetItalic( true ); } m_fields->push_back( newField ); // notify the grid wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 ); m_grid->ProcessTableMessage( msg ); m_grid->MakeCellVisible( m_fields->size() - 1, 0 ); m_grid->SetGridCursor( m_fields->size() - 1, fieldName == wxT( "Netclass" ) ? 1 : 0 ); m_grid->EnableCellEditControl(); m_grid->ShowCellEditControl(); } void DIALOG_LABEL_PROPERTIES::OnDeleteField( wxCommandEvent& event ) { wxArrayInt selectedRows = m_grid->GetSelectedRows(); if( selectedRows.empty() && m_grid->GetGridCursorRow() >= 0 ) selectedRows.push_back( m_grid->GetGridCursorRow() ); if( selectedRows.empty() ) return; for( int row : selectedRows ) { if( row < m_currentLabel->GetMandatoryFieldCount() ) { DisplayError( this, _( "The first field is mandatory." ) ); return; } } m_grid->CommitPendingChanges( true /* quiet mode */ ); // Reverse sort so deleting a row doesn't change the indexes of the other rows. selectedRows.Sort( []( int* first, int* second ) { return *second - *first; } ); for( int row : selectedRows ) { m_fields->erase( m_fields->begin() + row ); // notify the grid wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_DELETED, row, 1 ); m_grid->ProcessTableMessage( msg ); if( m_grid->GetNumberRows() > 0 ) { m_grid->MakeCellVisible( std::max( 0, row-1 ), m_grid->GetGridCursorCol() ); m_grid->SetGridCursor( std::max( 0, row-1 ), m_grid->GetGridCursorCol() ); } } } void DIALOG_LABEL_PROPERTIES::OnMoveUp( wxCommandEvent& event ) { if( !m_grid->CommitPendingChanges() ) return; int i = m_grid->GetGridCursorRow(); if( i > m_currentLabel->GetMandatoryFieldCount() ) { SCH_FIELD tmp = m_fields->at( (unsigned) i ); m_fields->erase( m_fields->begin() + i, m_fields->begin() + i + 1 ); m_fields->insert( m_fields->begin() + i - 1, tmp ); m_grid->ForceRefresh(); m_grid->SetGridCursor( i - 1, m_grid->GetGridCursorCol() ); m_grid->MakeCellVisible( m_grid->GetGridCursorRow(), m_grid->GetGridCursorCol() ); } else { wxBell(); } } void DIALOG_LABEL_PROPERTIES::OnMoveDown( wxCommandEvent& event ) { if( !m_grid->CommitPendingChanges() ) return; int i = m_grid->GetGridCursorRow(); if( i >= m_currentLabel->GetMandatoryFieldCount() && i < m_grid->GetNumberRows() - 1 ) { SCH_FIELD tmp = m_fields->at( (unsigned) i ); m_fields->erase( m_fields->begin() + i, m_fields->begin() + i + 1 ); m_fields->insert( m_fields->begin() + i + 1, tmp ); m_grid->ForceRefresh(); m_grid->SetGridCursor( i + 1, m_grid->GetGridCursorCol() ); m_grid->MakeCellVisible( m_grid->GetGridCursorRow(), m_grid->GetGridCursorCol() ); } else { wxBell(); } } void DIALOG_LABEL_PROPERTIES::AdjustGridColumns( int aWidth ) { m_width = aWidth; // Account for scroll bars aWidth -= ( m_grid->GetSize().x - m_grid->GetClientSize().x ); m_grid->AutoSizeColumn( 0 ); m_grid->SetColSize( 0, std::max( m_grid->GetColSize( 0 ), 72 ) ); int fixedColsWidth = m_grid->GetColSize( 0 ); for( int i = 2; i < m_grid->GetNumberCols(); i++ ) fixedColsWidth += m_grid->GetColSize( i ); m_grid->SetColSize( 1, std::max( aWidth - fixedColsWidth, 120 ) ); } void DIALOG_LABEL_PROPERTIES::OnUpdateUI( wxUpdateUIEvent& event ) { wxString shownColumns = m_grid->GetShownColumns(); if( shownColumns != m_shownColumns ) { m_shownColumns = shownColumns; if( !m_grid->IsCellEditControlShown() ) AdjustGridColumns( m_grid->GetRect().GetWidth() ); } // Handle a delayed focus if( m_delayedFocusRow >= 0 && m_delayedFocusRow < m_grid->GetNumberRows() ) { m_grid->SetFocus(); m_grid->MakeCellVisible( m_delayedFocusRow, m_delayedFocusColumn ); m_grid->SetGridCursor( m_delayedFocusRow, m_delayedFocusColumn ); m_grid->EnableCellEditControl( true ); m_grid->ShowCellEditControl(); } m_delayedFocusRow = -1; m_delayedFocusColumn = -1; } void DIALOG_LABEL_PROPERTIES::OnSizeGrid( wxSizeEvent& event ) { int new_size = event.GetSize().GetX(); if( m_width != new_size ) AdjustGridColumns( new_size ); // Always propagate for a grid repaint (needed if the height changes, as well as width) event.Skip(); }