A chave aqui é configurar o arquivo router.php dos componentes (que deve ser encontrado na pasta raiz do seu componente no front end) com uma lógica que irá procurar e selecionar o item de menu apropriado. Eu adoraria ver isso acontecer automaticamente, mas até onde eu sei, esse não é o caso.
Provavelmente, seria melhor trabalhar esse bloco de código em algum tipo de função auxiliar que possa ser usada para encontrar automaticamente o item de menu mais adequado ao conteúdo.
Aqui está o código que eu usei em vários dos meus componentes personalizados para obter o item de menu mais adequado:
// I use this first empty array to avoid having unset properties in my query
$base_array = array('Itemid'=>'', 'option'=>'', 'view'=>'', 'layout'=>'', 'id'=>'');
$app =& JFactory::getApplication();
$menu = $app->getMenu();
$active = $menu->getActive();
// hack to protect the actual current item as well as the search module or other places that use JRoute::_('index.php');
if (count($query)==2 && isset($query['option']) && isset($query['Itemid']) && $query['option'] && $query['Itemid']) {
return $segments;
}
// start with no match found
$match = false;
$match_level = 0;
$query += $base_array;
// we want to find a menu item for this if possible. If the active menu item is the current menu item then we should see if there is a better match.
if (empty($query['Itemid']) || ($query['Itemid'] == $active->id && empty($query['task']))) {
// load all menu items
$items = $menu->getMenu();
// use the current item over others if it ties for the best match
if ($active->query['option'] == $query['option']) {
$match_level = 1;
$match = $active;
if ($active->query['view'] == $query['view']) {
$match_level = 2;
if ($active->query['layout'] == $query['layout'] || ($query['layout']=='default' && !$active->query['layout'])) {
$match_level = 3;
if ($active->query['id'] == $query['id']) {
$match_level = 4;
}
}
}
}
// loop through each menu item in order
foreach ($items as $item) {
$item->query += $base_array;
// base check is that it is for this component
// then cycle through each possibility finding it's match level
if ($item->query['option'] == $query['option']) {
$item_match = 1;
if ($item->query['view'] == $query['view']) {
$item_match = 2;
if (!$query['layout'] && $item->query['layout']) {
$query['layout'] = 'default';
}
if ($item->query['layout'] == $query['layout'] || ($query['layout']=='default' && !$item->query['layout'])) {
$item_match = 3;
if ($item->query['id'] == $query['id']) {
$item_match = 4;
}
}
}
}
// if this item is a better match than our current match, set it as the best match
if ($item_match > $match_level) {
$match = $item;
$match_level = $item_match;
}
}
// if there is a match update Itemid to match that menu item
if ($match) {
$query['Itemid'] = $match->id;
$menuItem = $menu->getItem($match->id);
} else {
$menuItem = $menu->getActive();
}
}
Tudo isso é uma bagunça (e eu adoraria melhorias se alguém as tiver!), Mas faz o trabalho. Se o item de menu atual for a melhor correspondência, ele sempre permanecerá com isso.
Caso contrário, ele deve encontrar a melhor correspondência com base em Nome do componente -> nome da exibição -> nome do layout -> valor do ID. Quanto mais à direita corresponder, melhor considero a partida!