Add basic VsCodeThemeConverter
This commit is contained in:
parent
e48332c81f
commit
0bc51382b2
4 changed files with 92 additions and 82 deletions
|
@ -4,14 +4,17 @@
|
||||||
"themes": [
|
"themes": [
|
||||||
{
|
{
|
||||||
"name": "Ayu Light",
|
"name": "Ayu Light",
|
||||||
|
"file_name": "ayu-light.json",
|
||||||
"appearance": "light"
|
"appearance": "light"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Ayu Mirage",
|
"name": "Ayu Mirage",
|
||||||
|
"file_name": "ayu-mirage.json",
|
||||||
"appearance": "dark"
|
"appearance": "dark"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Ayu Dark",
|
"name": "Ayu Dark",
|
||||||
|
"file_name": "ayu-dark.json",
|
||||||
"appearance": "dark"
|
"appearance": "dark"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
"themes": [
|
"themes": [
|
||||||
{
|
{
|
||||||
"name": "Dracula",
|
"name": "Dracula",
|
||||||
|
"file_name": "dracula.json",
|
||||||
"appearance": "dark"
|
"appearance": "dark"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
|
@ -1,19 +1,17 @@
|
||||||
use std::borrow::Cow;
|
|
||||||
use std::fs::{self, File};
|
use std::fs::{self, File};
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use anyhow::{anyhow, Context, Result};
|
use anyhow::{anyhow, Context, Result};
|
||||||
use convert_case::Case;
|
use gpui::serde_json;
|
||||||
use gpui::{serde_json, AssetSource, SharedString};
|
|
||||||
use log::LevelFilter;
|
use log::LevelFilter;
|
||||||
use rust_embed::RustEmbed;
|
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use simplelog::SimpleLogger;
|
use simplelog::SimpleLogger;
|
||||||
use theme::{
|
use theme::{
|
||||||
default_color_scales, Appearance, GitStatusColors, PlayerColors, StatusColors, SyntaxTheme,
|
default_color_scales, Appearance, GitStatusColors, PlayerColors, StatusColors, SyntaxTheme,
|
||||||
SystemColors, ThemeColors, ThemeColorsRefinement, ThemeFamily, ThemeStyles, ThemeVariant,
|
SystemColors, ThemeColors, ThemeFamily, ThemeStyles, ThemeVariant,
|
||||||
};
|
};
|
||||||
|
use vscode::VsCodeThemeConverter;
|
||||||
|
|
||||||
use crate::vscode::VsCodeTheme;
|
use crate::vscode::VsCodeTheme;
|
||||||
|
|
||||||
|
@ -30,10 +28,10 @@ pub(crate) fn new_theme_family(name: String, author: String) -> ThemeFamily {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct FamilyJson {
|
struct FamilyMetadata {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub author: String,
|
pub author: String,
|
||||||
pub themes: Vec<ThemeVariantJson>,
|
pub themes: Vec<ThemeMetadata>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
#[derive(Debug, Deserialize)]
|
||||||
|
@ -43,28 +41,22 @@ enum ThemeAppearanceJson {
|
||||||
Dark,
|
Dark,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<ThemeAppearanceJson> for Appearance {
|
||||||
|
fn from(value: ThemeAppearanceJson) -> Self {
|
||||||
|
match value {
|
||||||
|
ThemeAppearanceJson::Light => Self::Light,
|
||||||
|
ThemeAppearanceJson::Dark => Self::Dark,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct ThemeVariantJson {
|
struct ThemeMetadata {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
|
pub file_name: String,
|
||||||
pub appearance: ThemeAppearanceJson,
|
pub appearance: ThemeAppearanceJson,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ImportedThemeFamily {
|
|
||||||
pub id: String,
|
|
||||||
pub name: String,
|
|
||||||
pub author: String,
|
|
||||||
pub url: Option<String>,
|
|
||||||
// App should panic if we try to load a theme without a lisence
|
|
||||||
pub license: String,
|
|
||||||
pub themes: Vec<ImportedThemeVariant>,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ImportedThemeVariant {
|
|
||||||
pub id: String,
|
|
||||||
pub name: String,
|
|
||||||
pub colors: ThemeColorsRefinement,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load a vscode theme from json
|
// Load a vscode theme from json
|
||||||
// Load it's LICENSE from the same folder
|
// Load it's LICENSE from the same folder
|
||||||
// Create a ThemeFamily for the theme
|
// Create a ThemeFamily for the theme
|
||||||
|
@ -92,35 +84,25 @@ fn main() -> Result<()> {
|
||||||
let family_metadata_file = File::open(theme_family_dir.path().join("family.json"))
|
let family_metadata_file = File::open(theme_family_dir.path().join("family.json"))
|
||||||
.context(format!("no `family.json` found for '{theme_family_slug}'"))?;
|
.context(format!("no `family.json` found for '{theme_family_slug}'"))?;
|
||||||
|
|
||||||
let family_metadata: FamilyJson = serde_json::from_reader(family_metadata_file).context(
|
let family_metadata: FamilyMetadata = serde_json::from_reader(family_metadata_file)
|
||||||
format!("failed to parse `family.json` for '{theme_family_slug}'"),
|
.context(format!(
|
||||||
)?;
|
"failed to parse `family.json` for '{theme_family_slug}'"
|
||||||
|
))?;
|
||||||
|
|
||||||
let mut themes = Vec::new();
|
let mut themes = Vec::new();
|
||||||
|
|
||||||
for theme_entry in fs::read_dir(vscode_themes_path.join(theme_family_slug))? {
|
for theme_metadata in family_metadata.themes {
|
||||||
let theme_entry = theme_entry?;
|
let theme_file_path = theme_family_dir.path().join(&theme_metadata.file_name);
|
||||||
|
|
||||||
let theme_file_path = theme_entry.path();
|
|
||||||
|
|
||||||
let file_name = theme_file_path
|
|
||||||
.file_name()
|
|
||||||
.ok_or(anyhow!("no file stem"))
|
|
||||||
.map(|file_name| file_name.to_string_lossy())?;
|
|
||||||
|
|
||||||
if !file_name.ends_with(".json") {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if file_name == "family.json" {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let theme_file = File::open(&theme_file_path)?;
|
let theme_file = File::open(&theme_file_path)?;
|
||||||
|
|
||||||
let theme: VsCodeTheme = serde_json::from_reader(theme_file)
|
let vscode_theme: VsCodeTheme = serde_json::from_reader(theme_file)
|
||||||
.context(format!("failed to parse theme {theme_file_path:?}"))?;
|
.context(format!("failed to parse theme {theme_file_path:?}"))?;
|
||||||
|
|
||||||
|
let converter = VsCodeThemeConverter::new(vscode_theme, theme_metadata);
|
||||||
|
|
||||||
|
let theme = converter.convert()?;
|
||||||
|
|
||||||
themes.push(theme);
|
themes.push(theme);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -128,22 +110,7 @@ fn main() -> Result<()> {
|
||||||
id: uuid::Uuid::new_v4().to_string(),
|
id: uuid::Uuid::new_v4().to_string(),
|
||||||
name: family_metadata.name.into(),
|
name: family_metadata.name.into(),
|
||||||
author: family_metadata.author.into(),
|
author: family_metadata.author.into(),
|
||||||
themes: themes
|
themes,
|
||||||
.into_iter()
|
|
||||||
.map(|theme| ThemeVariant {
|
|
||||||
id: uuid::Uuid::new_v4().to_string(),
|
|
||||||
name: "".into(),
|
|
||||||
appearance: Appearance::Dark,
|
|
||||||
styles: ThemeStyles {
|
|
||||||
system: SystemColors::default(),
|
|
||||||
colors: ThemeColors::default_dark(),
|
|
||||||
status: StatusColors::default(),
|
|
||||||
git: GitStatusColors::default(),
|
|
||||||
player: PlayerColors::default(),
|
|
||||||
syntax: SyntaxTheme::default_dark(),
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.collect(),
|
|
||||||
scales: default_color_scales(),
|
scales: default_color_scales(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -152,23 +119,3 @@ fn main() -> Result<()> {
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(RustEmbed)]
|
|
||||||
#[folder = "../../assets"]
|
|
||||||
#[include = "themes/**/*"]
|
|
||||||
pub struct Assets;
|
|
||||||
|
|
||||||
impl AssetSource for Assets {
|
|
||||||
fn load(&self, path: &str) -> Result<Cow<[u8]>> {
|
|
||||||
Self::get(path)
|
|
||||||
.map(|f| f.data)
|
|
||||||
.ok_or_else(|| anyhow!("could not find asset at path \"{}\"", path))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn list(&self, path: &str) -> Result<Vec<SharedString>> {
|
|
||||||
Ok(Self::iter()
|
|
||||||
.filter(|p| p.starts_with(path))
|
|
||||||
.map(SharedString::from)
|
|
||||||
.collect())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,8 +1,16 @@
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
use gpui::{Hsla, Refineable, Rgba};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use theme::{default_color_scales, ColorScales, ThemeFamily};
|
use theme::{
|
||||||
|
default_color_scales, Appearance, ColorScales, GitStatusColors, PlayerColors, StatusColors,
|
||||||
|
SyntaxTheme, SystemColors, ThemeColors, ThemeColorsRefinement, ThemeFamily, ThemeStyles,
|
||||||
|
ThemeVariant,
|
||||||
|
};
|
||||||
|
use uuid::Uuid;
|
||||||
|
|
||||||
|
use crate::ThemeMetadata;
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Deserialize, Debug)]
|
||||||
pub struct VsCodeTheme {
|
pub struct VsCodeTheme {
|
||||||
|
@ -61,6 +69,57 @@ pub(crate) fn new_theme_family_from_vsc(path: &Path) -> Result<ThemeFamily> {
|
||||||
// Ok(theme_family)
|
// Ok(theme_family)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn try_parse_color(color: &str) -> Result<Hsla> {
|
||||||
|
Ok(Rgba::try_from(color)?.into())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct VsCodeThemeConverter {
|
||||||
|
theme: VsCodeTheme,
|
||||||
|
theme_metadata: ThemeMetadata,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl VsCodeThemeConverter {
|
||||||
|
pub fn new(theme: VsCodeTheme, theme_metadata: ThemeMetadata) -> Self {
|
||||||
|
Self {
|
||||||
|
theme,
|
||||||
|
theme_metadata,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn convert(self) -> Result<ThemeVariant> {
|
||||||
|
let appearance = self.theme_metadata.appearance.into();
|
||||||
|
|
||||||
|
let mut theme_colors = match appearance {
|
||||||
|
Appearance::Light => ThemeColors::default_light(),
|
||||||
|
Appearance::Dark => ThemeColors::default_dark(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let vscode_colors = &self.theme.colors;
|
||||||
|
|
||||||
|
let theme_colors_refinements = ThemeColorsRefinement {
|
||||||
|
background: Some(try_parse_color(&vscode_colors.editor)?),
|
||||||
|
text: Some(try_parse_color(&vscode_colors.text)?),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
|
||||||
|
theme_colors.refine(&theme_colors_refinements);
|
||||||
|
|
||||||
|
Ok(ThemeVariant {
|
||||||
|
id: uuid::Uuid::new_v4().to_string(),
|
||||||
|
name: self.theme_metadata.name.into(),
|
||||||
|
appearance,
|
||||||
|
styles: ThemeStyles {
|
||||||
|
system: SystemColors::default(),
|
||||||
|
colors: theme_colors,
|
||||||
|
status: StatusColors::default(),
|
||||||
|
git: GitStatusColors::default(),
|
||||||
|
player: PlayerColors::default(),
|
||||||
|
syntax: SyntaxTheme::default_dark(),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue