use std::collections::HashMap;
use glua_code_analysis::{
LuaCompilation, LuaMemberId, LuaSemanticDeclId, SemanticDeclLevel, SemanticModel,
};
use glua_parser::{
LuaAst, LuaAstNode, LuaAstToken, LuaIndexToken, LuaLiteralExpr, LuaNameToken, LuaNumberToken,
LuaSyntaxNode, LuaTokenKind,
};
use lsp_types::Uri;
use crate::handlers::hover::find_member_origin_owner;
#[allow(clippy::mutable_key_type)]
pub fn rename_member_references(
semantic_model: &SemanticModel,
compilation: &LuaCompilation,
member_id: LuaMemberId,
new_name: String,
result: &mut HashMap<Uri, HashMap<lsp_types::Range, String>>,
) -> Option<()> {
let member = semantic_model
.get_db()
.get_member_index()
.get_member(&member_id)?;
let key = member.get_key();
let index_references: Vec<glua_code_analysis::InFiled<glua_parser::LuaSyntaxId>> =
semantic_model
.get_db()
.get_reference_index()
.get_index_references(key)?;
let origin_property_owner = find_member_origin_owner(compilation, semantic_model, member_id)
.unwrap_or(LuaSemanticDeclId::Member(member_id));
let property_owner = LuaSemanticDeclId::Member(member_id);
let mut semantic_cache = HashMap::new();
for in_filed_syntax_id in index_references {
let semantic_model =
if let Some(semantic_model) = semantic_cache.get_mut(&in_filed_syntax_id.file_id) {
semantic_model
} else {
let semantic_model = compilation.get_semantic_model(in_filed_syntax_id.file_id)?;
semantic_cache.insert(in_filed_syntax_id.file_id, semantic_model);
semantic_cache.get_mut(&in_filed_syntax_id.file_id)?
};
let root = semantic_model.get_root();
let node = in_filed_syntax_id.value.to_node_from_root(root.syntax())?;
if semantic_model.is_reference_to(
node.clone(),
origin_property_owner.clone(),
SemanticDeclLevel::NoTrace,
) || semantic_model.is_reference_to(
node.clone(),
property_owner.clone(),
SemanticDeclLevel::NoTrace,
) {
let range = get_member_name_token_lsp_range(semantic_model, node.clone());
if let Some(range) = range {
result
.entry(semantic_model.get_document().get_uri())
.or_default()
.insert(range, new_name.clone());
}
}
}
Some(())
}
fn get_member_name_token_lsp_range(
semantic_model: &SemanticModel,
node: LuaSyntaxNode,
) -> Option<lsp_types::Range> {
let document = semantic_model.get_document();
let node = LuaAst::cast(node)?;
if let Some(token) = node.token::<LuaNameToken>() {
return document.to_lsp_range(token.get_range());
}
if node.token::<LuaIndexToken>().is_some() {
match node {
LuaAst::LuaDocTagField(tag) => {
if let Some(token) = tag.token::<LuaNumberToken>() {
if matches!(token.get_token_kind(), LuaTokenKind::TkInt) {
return document.to_lsp_range(token.get_range());
}
return document.to_lsp_range(token.get_range());
}
}
_ => {
let literal = node.child::<LuaLiteralExpr>()?.get_literal()?;
if matches!(
literal.get_token_kind(),
LuaTokenKind::TkInt | LuaTokenKind::TkString
) {
return document.to_lsp_range(literal.get_range());
}
}
}
}
None
}