2014-06-25 16:29:58 +00:00
# include "components/ScraperSearchComponent.h"
2017-11-01 22:21:10 +00:00
# include "components/ComponentList.h"
2018-10-13 01:08:15 +00:00
# include "components/DateTimeEditComponent.h"
2014-06-25 16:29:58 +00:00
# include "components/ImageComponent.h"
# include "components/RatingComponent.h"
2017-11-01 22:21:10 +00:00
# include "components/ScrollableContainer.h"
# include "components/TextComponent.h"
# include "guis/GuiMsgBox.h"
# include "guis/GuiTextEditPopup.h"
# include "resources/Font.h"
2018-01-27 17:04:28 +00:00
# include "utils/StringUtil.h"
2017-11-01 22:21:10 +00:00
# include "FileData.h"
2014-06-25 16:29:58 +00:00
# include "Log.h"
2017-11-01 22:21:10 +00:00
# include "Window.h"
2014-06-25 16:29:58 +00:00
ScraperSearchComponent : : ScraperSearchComponent ( Window * window , SearchType type ) : GuiComponent ( window ) ,
2019-08-25 15:23:02 +00:00
mGrid ( window , Vector2i ( 4 , 3 ) ) , mBusyAnim ( window ) ,
2014-06-25 16:29:58 +00:00
mSearchType ( type )
{
addChild ( & mGrid ) ;
mBlockAccept = false ;
// left spacer (empty component, needed for borders)
mGrid . setEntry ( std : : make_shared < GuiComponent > ( mWindow ) , Vector2i ( 0 , 0 ) , false , false , Vector2i ( 1 , 3 ) , GridFlags : : BORDER_TOP | GridFlags : : BORDER_BOTTOM ) ;
// selected result name
mResultName = std : : make_shared < TextComponent > ( mWindow , " Result name " , Font : : get ( FONT_SIZE_MEDIUM ) , 0x777777FF ) ;
// selected result thumbnail
mResultThumbnail = std : : make_shared < ImageComponent > ( mWindow ) ;
mGrid . setEntry ( mResultThumbnail , Vector2i ( 1 , 1 ) , false , false , Vector2i ( 1 , 1 ) ) ;
// selected result desc + container
mDescContainer = std : : make_shared < ScrollableContainer > ( mWindow ) ;
mResultDesc = std : : make_shared < TextComponent > ( mWindow , " Result desc " , Font : : get ( FONT_SIZE_SMALL ) , 0x777777FF ) ;
mDescContainer - > addChild ( mResultDesc . get ( ) ) ;
mDescContainer - > setAutoScroll ( true ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
// metadata
auto font = Font : : get ( FONT_SIZE_SMALL ) ; // this gets replaced in onSizeChanged() so its just a placeholder
const unsigned int mdColor = 0x777777FF ;
const unsigned int mdLblColor = 0x666666FF ;
mMD_Rating = std : : make_shared < RatingComponent > ( mWindow ) ;
2018-10-13 01:08:15 +00:00
mMD_ReleaseDate = std : : make_shared < DateTimeEditComponent > ( mWindow ) ;
2014-06-25 16:29:58 +00:00
mMD_ReleaseDate - > setColor ( mdColor ) ;
mMD_Developer = std : : make_shared < TextComponent > ( mWindow , " " , font , mdColor ) ;
mMD_Publisher = std : : make_shared < TextComponent > ( mWindow , " " , font , mdColor ) ;
mMD_Genre = std : : make_shared < TextComponent > ( mWindow , " " , font , mdColor ) ;
mMD_Players = std : : make_shared < TextComponent > ( mWindow , " " , font , mdColor ) ;
mMD_Pairs . push_back ( MetaDataPair ( std : : make_shared < TextComponent > ( mWindow , " RATING: " , font , mdLblColor ) , mMD_Rating , false ) ) ;
mMD_Pairs . push_back ( MetaDataPair ( std : : make_shared < TextComponent > ( mWindow , " RELEASED: " , font , mdLblColor ) , mMD_ReleaseDate ) ) ;
mMD_Pairs . push_back ( MetaDataPair ( std : : make_shared < TextComponent > ( mWindow , " DEVELOPER: " , font , mdLblColor ) , mMD_Developer ) ) ;
mMD_Pairs . push_back ( MetaDataPair ( std : : make_shared < TextComponent > ( mWindow , " PUBLISHER: " , font , mdLblColor ) , mMD_Publisher ) ) ;
mMD_Pairs . push_back ( MetaDataPair ( std : : make_shared < TextComponent > ( mWindow , " GENRE: " , font , mdLblColor ) , mMD_Genre ) ) ;
mMD_Pairs . push_back ( MetaDataPair ( std : : make_shared < TextComponent > ( mWindow , " PLAYERS: " , font , mdLblColor ) , mMD_Players ) ) ;
2017-11-17 14:58:52 +00:00
mMD_Grid = std : : make_shared < ComponentGrid > ( mWindow , Vector2i ( 2 , ( int ) mMD_Pairs . size ( ) * 2 - 1 ) ) ;
2014-06-25 16:29:58 +00:00
unsigned int i = 0 ;
2017-11-11 14:56:22 +00:00
for ( auto it = mMD_Pairs . cbegin ( ) ; it ! = mMD_Pairs . cend ( ) ; it + + )
2014-06-25 16:29:58 +00:00
{
mMD_Grid - > setEntry ( it - > first , Vector2i ( 0 , i ) , false , true ) ;
mMD_Grid - > setEntry ( it - > second , Vector2i ( 1 , i ) , false , it - > resize ) ;
i + = 2 ;
}
mGrid . setEntry ( mMD_Grid , Vector2i ( 2 , 1 ) , false , false ) ;
// result list
mResultList = std : : make_shared < ComponentList > ( mWindow ) ;
mResultList - > setCursorChangedCallback ( [ this ] ( CursorState state ) { if ( state = = CURSOR_STOPPED ) updateInfoPane ( ) ; } ) ;
updateViewStyle ( ) ;
}
void ScraperSearchComponent : : onSizeChanged ( )
{
mGrid . setSize ( mSize ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
if ( mSize . x ( ) = = 0 | | mSize . y ( ) = = 0 )
return ;
// column widths
if ( mSearchType = = ALWAYS_ACCEPT_FIRST_RESULT )
mGrid . setColWidthPerc ( 0 , 0.02f ) ; // looks better when this is higher in auto mode
else
mGrid . setColWidthPerc ( 0 , 0.01f ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
mGrid . setColWidthPerc ( 1 , 0.25f ) ;
mGrid . setColWidthPerc ( 2 , 0.25f ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
// row heights
if ( mSearchType = = ALWAYS_ACCEPT_FIRST_RESULT ) // show name
mGrid . setRowHeightPerc ( 0 , ( mResultName - > getFont ( ) - > getHeight ( ) * 1.6f ) / mGrid . getSize ( ) . y ( ) ) ; // result name
else
mGrid . setRowHeightPerc ( 0 , 0.0825f ) ; // hide name but do padding
if ( mSearchType = = ALWAYS_ACCEPT_FIRST_RESULT )
{
mGrid . setRowHeightPerc ( 2 , 0.2f ) ;
} else {
mGrid . setRowHeightPerc ( 1 , 0.505f ) ;
}
const float boxartCellScale = 0.9f ;
// limit thumbnail size using setMaxHeight - we do this instead of letting mGrid call setSize because it maintains the aspect ratio
// we also pad a little so it doesn't rub up against the metadata labels
mResultThumbnail - > setMaxSize ( mGrid . getColWidth ( 1 ) * boxartCellScale , mGrid . getRowHeight ( 1 ) ) ;
// metadata
resizeMetadata ( ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
if ( mSearchType ! = ALWAYS_ACCEPT_FIRST_RESULT )
mDescContainer - > setSize ( mGrid . getColWidth ( 1 ) * boxartCellScale + mGrid . getColWidth ( 2 ) , mResultDesc - > getFont ( ) - > getHeight ( ) * 3 ) ;
else
mDescContainer - > setSize ( mGrid . getColWidth ( 3 ) * boxartCellScale , mResultDesc - > getFont ( ) - > getHeight ( ) * 8 ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
mResultDesc - > setSize ( mDescContainer - > getSize ( ) . x ( ) , 0 ) ; // make desc text wrap at edge of container
mGrid . onSizeChanged ( ) ;
mBusyAnim . setSize ( mSize ) ;
}
void ScraperSearchComponent : : resizeMetadata ( )
{
mMD_Grid - > setSize ( mGrid . getColWidth ( 2 ) , mGrid . getRowHeight ( 1 ) ) ;
if ( mMD_Grid - > getSize ( ) . y ( ) > mMD_Pairs . size ( ) )
{
const int fontHeight = ( int ) ( mMD_Grid - > getSize ( ) . y ( ) / mMD_Pairs . size ( ) * 0.8f ) ;
auto fontLbl = Font : : get ( fontHeight , FONT_PATH_REGULAR ) ;
auto fontComp = Font : : get ( fontHeight , FONT_PATH_LIGHT ) ;
// update label fonts
float maxLblWidth = 0 ;
2017-11-11 14:56:22 +00:00
for ( auto it = mMD_Pairs . cbegin ( ) ; it ! = mMD_Pairs . cend ( ) ; it + + )
2014-06-25 16:29:58 +00:00
{
it - > first - > setFont ( fontLbl ) ;
it - > first - > setSize ( 0 , 0 ) ;
if ( it - > first - > getSize ( ) . x ( ) > maxLblWidth )
maxLblWidth = it - > first - > getSize ( ) . x ( ) + 6 ;
}
for ( unsigned int i = 0 ; i < mMD_Pairs . size ( ) ; i + + )
{
mMD_Grid - > setRowHeightPerc ( i * 2 , ( fontLbl - > getLetterHeight ( ) + 2 ) / mMD_Grid - > getSize ( ) . y ( ) ) ;
}
// update component fonts
mMD_ReleaseDate - > setFont ( fontComp ) ;
mMD_Developer - > setFont ( fontComp ) ;
mMD_Publisher - > setFont ( fontComp ) ;
mMD_Genre - > setFont ( fontComp ) ;
mMD_Players - > setFont ( fontComp ) ;
mMD_Grid - > setColWidthPerc ( 0 , maxLblWidth / mMD_Grid - > getSize ( ) . x ( ) ) ;
// rating is manually sized
mMD_Rating - > setSize ( mMD_Grid - > getColWidth ( 1 ) , fontLbl - > getHeight ( ) * 0.65f ) ;
mMD_Grid - > onSizeChanged ( ) ;
// make result font follow label font
mResultDesc - > setFont ( Font : : get ( fontHeight , FONT_PATH_REGULAR ) ) ;
}
}
void ScraperSearchComponent : : updateViewStyle ( )
{
// unlink description and result list and result name
mGrid . removeEntry ( mResultName ) ;
mGrid . removeEntry ( mResultDesc ) ;
mGrid . removeEntry ( mResultList ) ;
// add them back depending on search type
if ( mSearchType = = ALWAYS_ACCEPT_FIRST_RESULT )
{
// show name
mGrid . setEntry ( mResultName , Vector2i ( 1 , 0 ) , false , true , Vector2i ( 2 , 1 ) , GridFlags : : BORDER_TOP ) ;
// need a border on the bottom left
mGrid . setEntry ( std : : make_shared < GuiComponent > ( mWindow ) , Vector2i ( 0 , 2 ) , false , false , Vector2i ( 3 , 1 ) , GridFlags : : BORDER_BOTTOM ) ;
// show description on the right
mGrid . setEntry ( mDescContainer , Vector2i ( 3 , 0 ) , false , false , Vector2i ( 1 , 3 ) , GridFlags : : BORDER_TOP | GridFlags : : BORDER_BOTTOM ) ;
mResultDesc - > setSize ( mDescContainer - > getSize ( ) . x ( ) , 0 ) ; // make desc text wrap at edge of container
} else {
// fake row where name would be
mGrid . setEntry ( std : : make_shared < GuiComponent > ( mWindow ) , Vector2i ( 1 , 0 ) , false , true , Vector2i ( 2 , 1 ) , GridFlags : : BORDER_TOP ) ;
// show result list on the right
mGrid . setEntry ( mResultList , Vector2i ( 3 , 0 ) , true , true , Vector2i ( 1 , 3 ) , GridFlags : : BORDER_LEFT | GridFlags : : BORDER_TOP | GridFlags : : BORDER_BOTTOM ) ;
// show description under image/info
mGrid . setEntry ( mDescContainer , Vector2i ( 1 , 2 ) , false , false , Vector2i ( 2 , 1 ) , GridFlags : : BORDER_BOTTOM ) ;
mResultDesc - > setSize ( mDescContainer - > getSize ( ) . x ( ) , 0 ) ; // make desc text wrap at edge of container
}
}
void ScraperSearchComponent : : search ( const ScraperSearchParams & params )
{
2017-05-25 17:56:06 +00:00
mBlockAccept = true ;
2014-06-25 16:29:58 +00:00
mResultList - > clear ( ) ;
mScraperResults . clear ( ) ;
mThumbnailReq . reset ( ) ;
mMDResolveHandle . reset ( ) ;
updateInfoPane ( ) ;
mLastSearch = params ;
mSearchHandle = startScraperSearch ( params ) ;
}
void ScraperSearchComponent : : stop ( )
{
mThumbnailReq . reset ( ) ;
mSearchHandle . reset ( ) ;
mMDResolveHandle . reset ( ) ;
mBlockAccept = false ;
}
void ScraperSearchComponent : : onSearchDone ( const std : : vector < ScraperSearchResult > & results )
{
mResultList - > clear ( ) ;
mScraperResults = results ;
auto font = Font : : get ( FONT_SIZE_MEDIUM ) ;
unsigned int color = 0x777777FF ;
2017-09-03 19:11:38 +00:00
if ( results . empty ( ) )
2014-06-25 16:29:58 +00:00
{
2019-01-24 18:00:19 +00:00
// Check if the scraper used is still valid
if ( ! isValidConfiguredScraper ( ) )
{
mWindow - > pushGui ( new GuiMsgBox ( mWindow , Utils : : String : : toUpper ( " Configured scraper is no longer available. \n Please change the scraping source in the settings. " ) ,
" FINISH " , mSkipCallback ) ) ;
}
else
{
ComponentListRow row ;
row . addElement ( std : : make_shared < TextComponent > ( mWindow , " NO GAMES FOUND - SKIP " , font , color ) , true ) ;
2014-06-25 16:29:58 +00:00
2019-01-24 18:00:19 +00:00
if ( mSkipCallback )
row . makeAcceptInputHandler ( mSkipCallback ) ;
2014-06-25 16:29:58 +00:00
2019-01-24 18:00:19 +00:00
mResultList - > addRow ( row ) ;
mGrid . resetCursor ( ) ;
}
2014-06-25 16:29:58 +00:00
} else {
ComponentListRow row ;
2017-09-03 19:11:38 +00:00
for ( size_t i = 0 ; i < results . size ( ) ; i + + )
2014-06-25 16:29:58 +00:00
{
row . elements . clear ( ) ;
2018-01-27 17:04:28 +00:00
row . addElement ( std : : make_shared < TextComponent > ( mWindow , Utils : : String : : toUpper ( results . at ( i ) . mdl . get ( " name " ) ) , font , color ) , true ) ;
2014-06-25 16:29:58 +00:00
row . makeAcceptInputHandler ( [ this , i ] { returnResult ( mScraperResults . at ( i ) ) ; } ) ;
mResultList - > addRow ( row ) ;
}
mGrid . resetCursor ( ) ;
}
mBlockAccept = false ;
updateInfoPane ( ) ;
if ( mSearchType = = ALWAYS_ACCEPT_FIRST_RESULT )
{
if ( mScraperResults . size ( ) = = 0 )
mSkipCallback ( ) ;
else
returnResult ( mScraperResults . front ( ) ) ;
} else if ( mSearchType = = ALWAYS_ACCEPT_MATCHING_CRC )
{
// TODO
}
}
void ScraperSearchComponent : : onSearchError ( const std : : string & error )
{
LOG ( LogInfo ) < < " ScraperSearchComponent search error: " < < error ;
2018-01-27 17:04:28 +00:00
mWindow - > pushGui ( new GuiMsgBox ( mWindow , Utils : : String : : toUpper ( error ) ,
2014-06-25 16:29:58 +00:00
" RETRY " , std : : bind ( & ScraperSearchComponent : : search , this , mLastSearch ) ,
" SKIP " , mSkipCallback ,
" CANCEL " , mCancelCallback ) ) ;
}
int ScraperSearchComponent : : getSelectedIndex ( )
{
if ( ! mScraperResults . size ( ) | | mGrid . getSelectedComponent ( ) ! = mResultList )
return - 1 ;
return mResultList - > getCursorId ( ) ;
}
void ScraperSearchComponent : : updateInfoPane ( )
{
int i = getSelectedIndex ( ) ;
if ( mSearchType = = ALWAYS_ACCEPT_FIRST_RESULT & & mScraperResults . size ( ) )
{
i = 0 ;
}
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
if ( i ! = - 1 & & ( int ) mScraperResults . size ( ) > i )
{
ScraperSearchResult & res = mScraperResults . at ( i ) ;
2018-01-27 17:04:28 +00:00
mResultName - > setText ( Utils : : String : : toUpper ( res . mdl . get ( " name " ) ) ) ;
mResultDesc - > setText ( Utils : : String : : toUpper ( res . mdl . get ( " desc " ) ) ) ;
2014-06-25 16:29:58 +00:00
mDescContainer - > reset ( ) ;
mResultThumbnail - > setImage ( " " ) ;
const std : : string & thumb = res . thumbnailUrl . empty ( ) ? res . imageUrl : res . thumbnailUrl ;
if ( ! thumb . empty ( ) )
{
mThumbnailReq = std : : unique_ptr < HttpReq > ( new HttpReq ( thumb ) ) ;
} else {
mThumbnailReq . reset ( ) ;
}
// metadata
2018-01-27 17:04:28 +00:00
mMD_Rating - > setValue ( Utils : : String : : toUpper ( res . mdl . get ( " rating " ) ) ) ;
mMD_ReleaseDate - > setValue ( Utils : : String : : toUpper ( res . mdl . get ( " releasedate " ) ) ) ;
mMD_Developer - > setText ( Utils : : String : : toUpper ( res . mdl . get ( " developer " ) ) ) ;
mMD_Publisher - > setText ( Utils : : String : : toUpper ( res . mdl . get ( " publisher " ) ) ) ;
mMD_Genre - > setText ( Utils : : String : : toUpper ( res . mdl . get ( " genre " ) ) ) ;
mMD_Players - > setText ( Utils : : String : : toUpper ( res . mdl . get ( " players " ) ) ) ;
2014-06-25 16:29:58 +00:00
mGrid . onSizeChanged ( ) ;
} else {
mResultName - > setText ( " " ) ;
mResultDesc - > setText ( " " ) ;
mResultThumbnail - > setImage ( " " ) ;
// metadata
mMD_Rating - > setValue ( " " ) ;
mMD_ReleaseDate - > setValue ( " " ) ;
mMD_Developer - > setText ( " " ) ;
mMD_Publisher - > setText ( " " ) ;
mMD_Genre - > setText ( " " ) ;
mMD_Players - > setText ( " " ) ;
}
}
bool ScraperSearchComponent : : input ( InputConfig * config , Input input )
{
if ( config - > isMappedTo ( " a " , input ) & & input . value ! = 0 )
{
if ( mBlockAccept )
return true ;
}
return GuiComponent : : input ( config , input ) ;
}
2017-10-28 20:24:35 +00:00
void ScraperSearchComponent : : render ( const Transform4x4f & parentTrans )
2014-06-25 16:29:58 +00:00
{
2017-10-28 20:24:35 +00:00
Transform4x4f trans = parentTrans * getTransform ( ) ;
2014-06-25 16:29:58 +00:00
renderChildren ( trans ) ;
if ( mBlockAccept )
{
Renderer : : setMatrix ( trans ) ;
2019-08-31 15:58:36 +00:00
Renderer : : drawRect ( 0.0f , 0.0f , mSize . x ( ) , mSize . y ( ) , 0x00000011 , 0x00000011 ) ;
2014-06-25 16:29:58 +00:00
mBusyAnim . render ( trans ) ;
}
}
void ScraperSearchComponent : : returnResult ( ScraperSearchResult result )
{
mBlockAccept = true ;
// resolve metadata image before returning
if ( ! result . imageUrl . empty ( ) )
{
mMDResolveHandle = resolveMetaDataAssets ( result , mLastSearch ) ;
return ;
}
mAcceptCallback ( result ) ;
}
void ScraperSearchComponent : : update ( int deltaTime )
{
GuiComponent : : update ( deltaTime ) ;
if ( mBlockAccept )
{
mBusyAnim . update ( deltaTime ) ;
}
if ( mThumbnailReq & & mThumbnailReq - > status ( ) ! = HttpReq : : REQ_IN_PROGRESS )
{
updateThumbnail ( ) ;
}
if ( mSearchHandle & & mSearchHandle - > status ( ) ! = ASYNC_IN_PROGRESS )
{
auto status = mSearchHandle - > status ( ) ;
auto results = mSearchHandle - > getResults ( ) ;
auto statusString = mSearchHandle - > getStatusString ( ) ;
2019-08-25 15:23:02 +00:00
// we reset here because onSearchDone in auto mode can call mSkipCallback() which can call
2014-06-25 16:29:58 +00:00
// another search() which will set our mSearchHandle to something important
mSearchHandle . reset ( ) ;
if ( status = = ASYNC_DONE )
{
onSearchDone ( results ) ;
} else if ( status = = ASYNC_ERROR )
{
onSearchError ( statusString ) ;
}
}
if ( mMDResolveHandle & & mMDResolveHandle - > status ( ) ! = ASYNC_IN_PROGRESS )
{
if ( mMDResolveHandle - > status ( ) = = ASYNC_DONE )
{
ScraperSearchResult result = mMDResolveHandle - > getResult ( ) ;
mMDResolveHandle . reset ( ) ;
// this might end in us being deleted, depending on mAcceptCallback - so make sure this is the last thing we do in update()
returnResult ( result ) ;
} else if ( mMDResolveHandle - > status ( ) = = ASYNC_ERROR )
{
onSearchError ( mMDResolveHandle - > getStatusString ( ) ) ;
mMDResolveHandle . reset ( ) ;
}
}
}
void ScraperSearchComponent : : updateThumbnail ( )
{
if ( mThumbnailReq & & mThumbnailReq - > status ( ) = = HttpReq : : REQ_SUCCESS )
{
std : : string content = mThumbnailReq - > getContent ( ) ;
mResultThumbnail - > setImage ( content . data ( ) , content . length ( ) ) ;
mGrid . onSizeChanged ( ) ; // a hack to fix the thumbnail position since its size changed
} else {
LOG ( LogWarning ) < < " thumbnail req failed: " < < mThumbnailReq - > getErrorMsg ( ) ;
mResultThumbnail - > setImage ( " " ) ;
}
mThumbnailReq . reset ( ) ;
}
void ScraperSearchComponent : : openInputScreen ( ScraperSearchParams & params )
{
auto searchForFunc = [ & ] ( const std : : string & name )
{
params . nameOverride = name ;
search ( params ) ;
} ;
stop ( ) ;
2019-08-25 15:23:02 +00:00
mWindow - > pushGui ( new GuiTextEditPopup ( mWindow , " SEARCH FOR " ,
2014-06-25 16:29:58 +00:00
// initial value is last search if there was one, otherwise the clean path name
2019-08-25 15:23:02 +00:00
params . nameOverride . empty ( ) ? params . game - > getCleanName ( ) : params . nameOverride ,
2014-06-25 16:29:58 +00:00
searchForFunc , false , " SEARCH " ) ) ;
}
std : : vector < HelpPrompt > ScraperSearchComponent : : getHelpPrompts ( )
{
std : : vector < HelpPrompt > prompts = mGrid . getHelpPrompts ( ) ;
if ( getSelectedIndex ( ) ! = - 1 )
prompts . push_back ( HelpPrompt ( " a " , " accept result " ) ) ;
2019-08-25 15:23:02 +00:00
2014-06-25 16:29:58 +00:00
return prompts ;
}
void ScraperSearchComponent : : onFocusGained ( )
{
mGrid . onFocusGained ( ) ;
}
void ScraperSearchComponent : : onFocusLost ( )
{
mGrid . onFocusLost ( ) ;
}