feat: Accept a file extension as the argument to :set-language

This commit is contained in:
emilylime 2025-03-11 16:45:05 +02:00
parent b38eae1f98
commit 9660916823
3 changed files with 24 additions and 14 deletions

View file

@ -959,16 +959,14 @@ impl Loader {
pub fn language_config_for_file_name(&self, path: &Path) -> Option<Arc<LanguageConfiguration>> { pub fn language_config_for_file_name(&self, path: &Path) -> Option<Arc<LanguageConfiguration>> {
// Find all the language configurations that match this file name // Find all the language configurations that match this file name
// or a suffix of the file name. // or a suffix of the file name.
let configuration_id = self self.language_config_ids_glob_matcher
.language_config_ids_glob_matcher
.language_id_for_path(path) .language_id_for_path(path)
.and_then(|&id| self.language_configs.get(id).cloned())
.or_else(|| { .or_else(|| {
path.extension() path.extension()
.and_then(|extension| extension.to_str()) .and_then(|extension| extension.to_str())
.and_then(|extension| self.language_config_ids_by_extension.get(extension)) .and_then(|extension| self.language_config_for_extension(extension))
}); })
configuration_id.and_then(|&id| self.language_configs.get(id).cloned())
// TODO: content_regex handling conflict resolution // TODO: content_regex handling conflict resolution
} }
@ -1000,10 +998,19 @@ impl Loader {
) -> Option<Arc<LanguageConfiguration>> { ) -> Option<Arc<LanguageConfiguration>> {
self.language_configs self.language_configs
.iter() .iter()
.find(|config| id.eq(&config.language_id)) .find(|config| id == config.language_id)
.cloned() .cloned()
} }
pub fn language_config_for_extension(
&self,
extension: &str,
) -> Option<Arc<LanguageConfiguration>> {
self.language_config_ids_by_extension
.get(extension)
.and_then(|&id| self.language_configs.get(id).cloned())
}
/// Unlike `language_config_for_language_id`, which only returns Some for an exact id, this /// Unlike `language_config_for_language_id`, which only returns Some for an exact id, this
/// function will perform a regex match on the given string to find the closest language match. /// function will perform a regex match on the given string to find the closest language match.
pub fn language_config_for_name(&self, slice: RopeSlice) -> Option<Arc<LanguageConfiguration>> { pub fn language_config_for_name(&self, slice: RopeSlice) -> Option<Arc<LanguageConfiguration>> {

View file

@ -2084,7 +2084,7 @@ fn language(cx: &mut compositor::Context, args: Args, event: PromptEvent) -> any
if &args[0] == DEFAULT_LANGUAGE_NAME { if &args[0] == DEFAULT_LANGUAGE_NAME {
doc.set_language(None, None) doc.set_language(None, None)
} else { } else {
doc.set_language_by_language_id(&args[0], cx.editor.syn_loader.clone())?; doc.set_language_by_human_name(&args[0], cx.editor.syn_loader.clone())?;
} }
doc.detect_indent_and_line_ending(); doc.detect_indent_and_line_ending();

View file

@ -1225,15 +1225,18 @@ impl Document {
/// Set the programming language for the file if you know the language but don't have the /// Set the programming language for the file if you know the language but don't have the
/// [`syntax::LanguageConfiguration`] for it. /// [`syntax::LanguageConfiguration`] for it.
pub fn set_language_by_language_id( pub fn set_language_by_human_name(
&mut self, &mut self,
language_id: &str, name: &str,
config_loader: Arc<ArcSwap<syntax::Loader>>, config_loader: Arc<ArcSwap<syntax::Loader>>,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let language_config = (*config_loader) let language_config = {
.load() let config = (*config_loader).load();
.language_config_for_language_id(language_id) config
.ok_or_else(|| anyhow!("invalid language id: {}", language_id))?; .language_config_for_language_id(name)
.or_else(|| config.language_config_for_extension(name))
.ok_or_else(|| anyhow!("unknown language: {}", name))?
};
self.set_language(Some(language_config), Some(config_loader)); self.set_language(Some(language_config), Some(config_loader));
Ok(()) Ok(())
} }