feat(models-ui): minor visual enhancements (#2109)

Show image if present, URL, tags, and better display buttons

Signed-off-by: Ettore Di Giacinto <mudler@localai.io>
This commit is contained in:
Ettore Di Giacinto 2024-04-23 18:43:25 +02:00 committed by GitHub
parent 3411e072ca
commit d344daf129
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 96 additions and 28 deletions

View File

@ -86,6 +86,18 @@ func StartProgressBar(uid, progress string) string {
).Render() ).Render()
} }
func cardSpan(text, icon string) elem.Node {
return elem.Span(
attrs.Props{
"class": "inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2",
},
elem.I(attrs.Props{
"class": icon + " pr-2",
}),
elem.Text(text),
)
}
func ListModels(models []*gallery.GalleryModel) string { func ListModels(models []*gallery.GalleryModel) string {
modelsElements := []elem.Node{} modelsElements := []elem.Node{}
span := func(s string) elem.Node { span := func(s string) elem.Node {
@ -99,10 +111,17 @@ func ListModels(models []*gallery.GalleryModel) string {
installButton := func(m *gallery.GalleryModel) elem.Node { installButton := func(m *gallery.GalleryModel) elem.Node {
return elem.Button( return elem.Button(
attrs.Props{ attrs.Props{
"class": "float-right inline-block rounded bg-primary px-6 pb-2 pt-2.5 text-xs font-medium uppercase leading-normal text-white shadow-primary-3 transition duration-150 ease-in-out hover:bg-primary-accent-300 hover:shadow-primary-2 focus:bg-primary-accent-300 focus:shadow-primary-2 focus:outline-none focus:ring-0 active:bg-primary-600 active:shadow-primary-2 dark:shadow-black/30 dark:hover:shadow-dark-strong dark:focus:shadow-dark-strong dark:active:shadow-dark-strong", "data-twe-ripple-init": "",
"data-twe-ripple-color": "light",
"class": "float-right inline-block rounded bg-primary px-6 pb-2.5 mb-3 pt-2.5 text-xs font-medium uppercase leading-normal text-white shadow-primary-3 transition duration-150 ease-in-out hover:bg-primary-accent-300 hover:shadow-primary-2 focus:bg-primary-accent-300 focus:shadow-primary-2 focus:outline-none focus:ring-0 active:bg-primary-600 active:shadow-primary-2 dark:shadow-black/30 dark:hover:shadow-dark-strong dark:focus:shadow-dark-strong dark:active:shadow-dark-strong",
// post the Model ID as param // post the Model ID as param
"hx-post": "/browse/install/model/" + fmt.Sprintf("%s@%s", m.Gallery.Name, m.Name), "hx-post": "/browse/install/model/" + fmt.Sprintf("%s@%s", m.Gallery.Name, m.Name),
}, },
elem.I(
attrs.Props{
"class": "fa-solid fa-download pr-2",
},
),
elem.Text("Install"), elem.Text("Install"),
) )
} }
@ -111,7 +130,7 @@ func ListModels(models []*gallery.GalleryModel) string {
return elem.Div( return elem.Div(
attrs.Props{ attrs.Props{
"class": "p-6", "class": "p-6 text-surface dark:text-white",
}, },
elem.H5( elem.H5(
attrs.Props{ attrs.Props{
@ -129,42 +148,93 @@ func ListModels(models []*gallery.GalleryModel) string {
} }
actionDiv := func(m *gallery.GalleryModel) elem.Node { actionDiv := func(m *gallery.GalleryModel) elem.Node {
nodes := []elem.Node{
cardSpan("Repository: "+m.Gallery.Name, "fa-brands fa-git-alt"),
}
if m.License != "" {
nodes = append(nodes,
cardSpan("License: "+m.License, "fas fa-book"),
)
}
for _, tag := range m.Tags {
nodes = append(nodes,
cardSpan(tag, "fas fa-tag"),
)
}
for i, url := range m.URLs {
nodes = append(nodes,
elem.A(
attrs.Props{
"class": "inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2",
"href": url,
"target": "_blank",
},
elem.I(attrs.Props{
"class": "fas fa-link pr-2",
}),
elem.Text("Link #"+fmt.Sprintf("%d", i+1)),
))
}
return elem.Div( return elem.Div(
attrs.Props{ attrs.Props{
"class": "px-6 pt-4 pb-2", "class": "px-6 pt-4 pb-2",
}, },
elem.Span( elem.P(
attrs.Props{ attrs.Props{
"class": "inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2", "class": "mb-4 text-base",
}, },
elem.Text("Repository: "+m.Gallery.Name), nodes...,
), ),
elem.If(m.Installed, span("Installed"), installButton(m)), elem.If(m.Installed, span("Installed"), installButton(m)),
) )
} }
for _, m := range models { for _, m := range models {
elems := []elem.Node{}
if m.Icon != "" {
elems = append(elems,
elem.Div(attrs.Props{
"class": "flex justify-center items-center",
},
elem.A(attrs.Props{
"href": "#!",
// "class": "justify-center items-center",
},
elem.Img(attrs.Props{
// "class": "rounded-t-lg object-fit object-center h-96",
"class": "rounded-t-lg max-h-48 max-w-96 object-cover",
"src": m.Icon,
}),
),
))
}
elems = append(elems, descriptionDiv(m), actionDiv(m))
modelsElements = append(modelsElements, modelsElements = append(modelsElements,
elem.Div( elem.Div(
attrs.Props{ attrs.Props{
"class": "me-4 mb-2 block rounded-lg bg-white shadow-secondary-1 dark:bg-gray-800 dark:bg-surface-dark dark:text-white text-surface p-2", "class": " me-4 mb-2 block rounded-lg bg-white shadow-secondary-1 dark:bg-gray-800 dark:bg-surface-dark dark:text-white text-surface pb-2",
}, },
elem.Div( elem.Div(
attrs.Props{ attrs.Props{
"class": "p-6", // "class": "p-6",
}, },
descriptionDiv(m), elems...,
actionDiv(m),
// elem.If(m.Installed, span("Installed"), installButton(m)),
// elem.If(m.Installed, span("Installed"), span("Not Installed")),
), ),
), ),
) )
} }
wrapper := elem.Div(attrs.Props{ wrapper := elem.Div(attrs.Props{
"class": "dark grid grid-cols-1 grid-rows-1 md:grid-cols-2 ", "class": "dark grid grid-cols-1 grid-rows-1 md:grid-cols-3 block rounded-lg shadow-secondary-1 dark:bg-surface-dark",
//"class": "block rounded-lg bg-white shadow-secondary-1 dark:bg-surface-dark",
}, modelsElements...) }, modelsElements...)
return wrapper.Render() return wrapper.Render()

View File

@ -26,8 +26,9 @@ func RegisterUIRoutes(app *fiber.App,
models, _ := gallery.AvailableGalleryModels(appConfig.Galleries, appConfig.ModelPath) models, _ := gallery.AvailableGalleryModels(appConfig.Galleries, appConfig.ModelPath)
summary := fiber.Map{ summary := fiber.Map{
"Title": "LocalAI API - Models", "Title": "LocalAI - Models",
"Models": template.HTML(elements.ListModels(models)), "Models": template.HTML(elements.ListModels(models)),
"Repositories": appConfig.Galleries,
// "ApplicationConfig": appConfig, // "ApplicationConfig": appConfig,
} }
@ -49,7 +50,10 @@ func RegisterUIRoutes(app *fiber.App,
filteredModels := []*gallery.GalleryModel{} filteredModels := []*gallery.GalleryModel{}
for _, m := range models { for _, m := range models {
if strings.Contains(m.Name, form.Search) { if strings.Contains(m.Name, form.Search) ||
strings.Contains(m.Description, form.Search) ||
strings.Contains(m.Gallery.Name, form.Search) ||
strings.Contains(strings.Join(m.Tags, ","), form.Search) {
filteredModels = append(filteredModels, m) filteredModels = append(filteredModels, m)
} }
} }

View File

@ -7,20 +7,14 @@
{{template "views/partials/navbar" .}} {{template "views/partials/navbar" .}}
<div class="container mx-auto px-4 flex-grow"> <div class="container mx-auto px-4 flex-grow">
<div class="header text-center py-12">
<h1 class="text-5xl font-bold text-gray-100">Welcome to <i>your</i> LocalAI instance!</h1>
<div class="mt-6">
<!-- Logo can be uncommented and updated with a valid URL -->
</div>
<p class="mt-4 text-lg">The FOSS alternative to OpenAI, Claude, ...</p>
<a href="https://localai.io" target="_blank" class="mt-4 inline-block bg-blue-500 text-white py-2 px-4 rounded-lg shadow transition duration-300 ease-in-out hover:bg-blue-700 hover:shadow-lg">
<i class="fas fa-book-reader pr-2"></i>Documentation
</a>
</div>
<div class="models mt-12"> <div class="models mt-12">
<h2 class="text-center text-3xl font-semibold text-gray-100">Available models from repositories</h2> <h2 class="text-center text-3xl font-semibold text-gray-100">
🖼️ Available models from <i>{{ len .Repositories }}</i> repositories <a href="https://localai.io/models/" target="_blank" >
<i class="fas fa-circle-info pr-2"></i>
</a></h2>
<span class="htmx-indicator loader"></span> <span class="htmx-indicator loader"></span>
<input class="form-control appearance-none block w-full px-3 py-2 text-base font-normal text-gray-300 pb-2 mb-5 bg-gray-800 bg-clip-padding border border-solid border-gray-600 rounded transition ease-in-out m-0 focus:text-gray-300 focus:bg-gray-900 focus:border-blue-500 focus:outline-none" type="search" <input class="form-control appearance-none block w-full px-3 py-2 text-base font-normal text-gray-300 pb-2 mb-5 bg-gray-800 bg-clip-padding border border-solid border-gray-600 rounded transition ease-in-out m-0 focus:text-gray-300 focus:bg-gray-900 focus:border-blue-500 focus:outline-none" type="search"
name="search" placeholder="Begin Typing To Search models..." name="search" placeholder="Begin Typing To Search models..."