docs/en/Community-Articles/2025-10-09-how-to-change-logo-in-angular-abp-apps/article.md
Logo application customization is one of the most common branding requirements in web applications. In ABP Framework's Angular applications, we found that developers were facing problems while they were trying to implement their application logos, especially on theme dependencies and flexibility. To overcome this, we moved the logo provider from @volo/ngx-lepton-x.core to @abp/ng.theme.shared, where it is more theme-independent and accessible. Here, we will describe our experience using this improvement and guide you on the new approach for logo configuration in ABP Angular applications.
Previously, the logo configuration process in ABP Angular applications had several disadvantages:
Theme Dependency: The provideLogo function was a part of the @volo/ngx-lepton-x.core package, so the developers had to depend on LeptonX theme packages even when they were using a different theme or wanted to extend the logo behavior.
Inflexibility: The fact that the logo provider had to adhere to a specific theme package brought about an undesirable tight coupling of logo configuration and theme implementation.
Discoverability Issues: Developers looking for logo configuration features would likely look in core ABP packages, but the provider was hidden in a theme-specific package, which made it harder to discover.
Migration Issues: During theme changes or theme package updates, logo setting could get corrupted or require additional tuning.
These made a basic operation like altering the application logo more challenging than it should be, especially for teams using custom themes or wanting to maintain theme independence.
We moved the provideLogo function from @volo/ngx-lepton-x.core to @abp/ng.theme.shared package. This solution offers:
This approach maintains ABP's philosophy of providing flexible, reusable solutions while reducing unnecessary dependencies.
Let's walk through how logo configuration works with the new approach.
First, define your logo URL in the environment.ts file:
export const environment = {
production: false,
application: {
baseUrl: 'http://localhost:4200',
name: 'MyApplication',
logoUrl: 'https://your-domain.com/assets/logo.png',
},
// ... other configurations
};
The logoUrl property accepts any valid URL, allowing you to use:
/assets/logo.png)In your app.config.ts (or app.module.ts for module-based apps), import and use the logo provider:
import { provideLogo, withEnvironmentOptions } from '@abp/ng.theme.shared';
import { environment } from './environments/environment';
export const appConfig: ApplicationConfig = {
providers: [
// ... other providers
provideLogo(withEnvironmentOptions(environment)),
],
};
Important Note: If you're migrating from an older version where the logo provider was in @volo/ngx-lepton-x.core, simply update the import statement:
// Old (before migration)
import { provideLogo, withEnvironmentOptions } from '@volo/ngx-lepton-x.core';
// New (current approach)
import { provideLogo, withEnvironmentOptions } from '@abp/ng.theme.shared';
The provideLogo function registers a logo configuration service that:
logoUrl from environment configurationThe withEnvironmentOptions helper extracts the relevant configuration from your environment object, ensuring type safety and proper configuration structure.
Here's a complete example showing both environment and provider configuration:
environment.ts:
export const environment = {
production: false,
application: {
baseUrl: 'http://localhost:4200',
name: 'E-Commerce Platform',
logoUrl: 'https://cdn.example.com/brand/logo-primary.svg',
},
oAuthConfig: {
issuer: 'https://localhost:44305',
clientId: 'MyApp_App',
// ... other OAuth settings
},
// ... other settings
};
app.config.ts:
import { ApplicationConfig } from '@angular/core';
import { provideRouter } from '@angular/router';
import { provideLogo, withEnvironmentOptions } from '@abp/ng.theme.shared';
import { environment } from './environments/environment';
import { routes } from './app.routes';
export const appConfig: ApplicationConfig = {
providers: [
provideRouter(routes),
provideLogo(withEnvironmentOptions(environment)),
// ... other providers
],
};
For more advanced customization scenarios where you need complete control over the logo component's structure, styling, or behavior, ABP provides a component replacement mechanism. This approach allows you to replace the entire logo component with your custom implementation.
Consider using component replacement when:
logoUrl configuration doesn't meet your requirementsRun the following command in your Angular folder to create a new component:
ng generate component custom-logo --inline-template --inline-style
Open the generated custom-logo.component.ts and implement your custom logo:
import { Component } from '@angular/core';
import { RouterModule } from '@angular/router';
@Component({
selector: 'app-custom-logo',
standalone: true,
imports: [RouterModule],
template: `
<a class="navbar-brand" routerLink="/">
</a>
`,
styles: [`
.navbar-brand {
padding: 0.5rem 1rem;
}
.navbar-brand img {
transition: opacity 0.3s ease;
}
.navbar-brand:hover img {
opacity: 0.8;
}
`]
})
export class CustomLogoComponent {}
Open your app.config.ts and register the component replacement:
import { ApplicationConfig } from '@angular/core';
import { provideRouter } from '@angular/router';
import { ReplaceableComponentsService } from '@abp/ng.core';
import { eThemeBasicComponents } from '@abp/ng.theme.basic';
import { CustomLogoComponent } from './custom-logo/custom-logo.component';
import { environment } from './environments/environment';
import { routes } from './app.routes';
export const appConfig: ApplicationConfig = {
providers: [
provideRouter(routes),
// ... other providers
{
provide: 'APP_INITIALIZER',
useFactory: (replaceableComponents: ReplaceableComponentsService) => {
return () => {
replaceableComponents.add({
component: CustomLogoComponent,
key: eThemeBasicComponents.Logo,
});
};
},
deps: [ReplaceableComponentsService],
multi: true,
},
],
};
Alternatively, if you're using a module-based application, you can register it in app.component.ts:
import { Component, OnInit } from '@angular/core';
import { ReplaceableComponentsService } from '@abp/ng.core';
import { eThemeBasicComponents } from '@abp/ng.theme.basic';
import { CustomLogoComponent } from './custom-logo/custom-logo.component';
@Component({
selector: 'app-root',
template: '<router-outlet></router-outlet>',
})
export class AppComponent implements OnInit {
constructor(private replaceableComponents: ReplaceableComponentsService) {}
ngOnInit() {
this.replaceableComponents.add({
component: CustomLogoComponent,
key: eThemeBasicComponents.Logo,
});
}
}
Here's a comparison to help you choose the right approach:
| Feature | Logo URL Configuration | Component Replacement |
|---|---|---|
| Simplicity | Very simple, one-line configuration | Requires creating a new component |
| Flexibility | Limited to image URL | Full control over HTML/CSS/behavior |
| Use Case | Standard logo display | Complex customizations |
| Maintenance | Minimal | Requires component maintenance |
| Migration | Easy to change | Requires code changes |
| Recommended For | Most applications | Advanced customization needs |
For most applications, the simple logoUrl configuration in the environment file is sufficient and recommended. Use component replacement only when you need advanced customization that goes beyond a simple image.
In this article, we explored how ABP Framework simplified logo configuration in Angular applications by moving the logo provider from @volo/ngx-lepton-x.core to @abp/ng.theme.shared. This change eliminates unnecessary theme dependencies and makes logo customization more straightforward and theme-agnostic.
The solution we implemented allows developers to configure their application logo simply by setting a URL in the environment file and providing the logo configuration in their application setup. For advanced scenarios requiring complete control over the logo component, ABP's component replacement mechanism provides a powerful alternative. This approach maintains flexibility while reducing complexity and improving discoverability.
We developed this improvement while working on ABP Framework to enhance developer experience and reduce common friction points. By sharing this solution, we hope to help teams implement consistent branding across their ABP Angular applications more easily, regardless of which theme they choose to use.
If you're using an older version of ABP with logo configuration in LeptonX packages, migrating to this new approach requires only a simple import path change, making it a smooth upgrade path for existing applications.