Skip to content

Commit

Permalink
[Home] configurable hero component and button
Browse files Browse the repository at this point in the history
Getting started image link if link is set.

Also, make it configurable.

Signed-off-by: Kawika Avilla <[email protected]>
  • Loading branch information
kavilla committed Nov 28, 2023
1 parent 162bed1 commit 26cfad2
Show file tree
Hide file tree
Showing 5 changed files with 149 additions and 77 deletions.
8 changes: 8 additions & 0 deletions config/opensearch_dashboards.yml
Original file line number Diff line number Diff line change
Expand Up @@ -273,3 +273,11 @@

# Set the value of this setting to true to enable plugin augmentation
# vis_augmenter.pluginAugmentationEnabled: true

# Example: For hero content
# home.hero.title: 'The Assistant'
# home.hero.body: 'Automatically make queries.{br}{br}All the {terms} applied.'
# home.hero.externalActionButton: {
# text: 'Login to try',
# link: 'https://observability.playground.opensearch.org/'
# }
100 changes: 65 additions & 35 deletions src/plugins/home/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,42 +34,72 @@ export const configSchema = schema.object({
disableWelcomeScreen: schema.boolean({ defaultValue: false }),
disableNewThemeModal: schema.boolean({ defaultValue: false }),
newHomepage: schema.boolean({ defaultValue: false }),
prompts: schema.arrayOf(
schema.object({
prompt: schema.string(),
indexPattern: schema.string(),
datasourceName: schema.string(),
datasourceType: schema.string(),
hero: schema.object({
enabled: schema.boolean({ defaultValue: true }),
title: schema.string({ defaultValue: 'Try the Query Assistant' }),
body: schema.string({
defaultValue:
'Automatically generate complex queries using simple conversational prompts. AI assisted summary helps you navigate and understand errors from your logs.{br}{br}You will be redirected to the observability playground where you will need to login. All the {terms} of the playground still apply.',
}),
{
defaultValue: [
{
prompt: 'How many errors are there in my logs?',
datasourceName: 'Default cluster',
datasourceType: 'DEFAULT_INDEX_PATTERNS',
indexPattern: 'sso-logs',
},
{
prompt: 'Show me the number of flights each day?',
datasourceName: 'Default cluster',
datasourceType: 'DEFAULT_INDEX_PATTERNS',
indexPattern: 'opensearch_dashboards_sample_data_flights',
},
{
prompt: 'What are top visited urls on my website?',
datasourceName: 'Default cluster',
datasourceType: 'DEFAULT_INDEX_PATTERNS',
indexPattern: 'opensearch_dashboards_sample_data_logs',
},
{
prompt: 'Show me the number of orders grouped by gender',
datasourceName: 'Default cluster',
datasourceType: 'DEFAULT_INDEX_PATTERNS',
indexPattern: 'opensearch_datashboards_sample_data_ecommerce',
},
],
}
),
img: schema.maybe(
schema.object({
src: schema.maybe(schema.string()),
link: schema.maybe(schema.string()),
})
),
actionButton: schema.object({
text: schema.string({ defaultValue: 'Try in Log Explorer' }),
app: schema.string({ defaultValue: 'observability-logs' }),
path: schema.string({ defaultValue: '#/explorer' }),
}),
externalActionButton: schema.maybe(
schema.object({
text: schema.string(),
link: schema.string(),
})
),
secondaryButton: schema.object({
text: schema.string({ defaultValue: 'Learn more' }),
link: schema.string({
defaultValue: 'https://opensearch.org/platform/observability/index.html',
}),
}),
prompts: schema.arrayOf(
schema.object({
text: schema.string(),
app: schema.string(),
path: schema.string(),
}),
{
defaultValue: [
{
text: 'How many errors are there in my logs?',
app: 'observability-logs',
path:
'#/explorer?datasourceType=DEFAULT_INDEX_PATTERNS&indexPattern=sso-logs&datasourceName=Default%20cluster',
},
{
text: 'Show me the number of flights each day?',
app: 'observability-logs',
path:
'#/explorer?datasourceType=DEFAULT_INDEX_PATTERNS&indexPattern=opensearch_dashboards_sample_data_flights&datasourceName=Default%20cluster',
},
{
text: 'What are top visited urls on my website?',
app: 'observability-logs',
path:
'#/explorer?datasourceType=DEFAULT_INDEX_PATTERNS&indexPattern=opensearch_dashboards_sample_data_logs&datasourceName=Default%20cluster',
},
{
text: 'Show me the number of orders grouped by gender',
app: 'observability-logs',
path:
'#/explorer?datasourceType=DEFAULT_INDEX_PATTERNS&indexPattern=opensearch_dashboards_sample_data_ecommerce&datasourceName=Default%20cluster',
},
],
}
),
}),
});

export type ConfigSchema = TypeOf<typeof configSchema>;
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,23 @@
.home-getStarted-chatIcon {
padding-left: $euiSizeXS;
}

.home-hero-illustrationContainer {
position: relative;
display: inline-block;
}

.home-hero-illustrationButton {
z-index: 1040;
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);

&:hover:not([class*="isDisabled"]),
&:active:not([class*="isDisabled"]),
&:focus:not([class*="isDisabled"]) {
transform: translate(-50%, -50%);
animation: none !important;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -15,31 +15,27 @@ import {
EuiImage,
EuiLink,
EuiIcon,
EuiButtonIcon,
} from '@elastic/eui';
import { HeroSection } from './hero_section';
import illustration from '../../../../assets/illustration.svg';
import { getServices } from '../../../opensearch_dashboards_services';
import logo from '../../../../assets/logos/chat.svg';
import image from '../../../../assets/screenshot.png';
import screenshot from '../../../../assets/screenshot.png';

export const GetStartedSection: React.FC<{ olly?: boolean }> = ({ olly = true }) => {
const services = getServices();
const getUrl = services.application.getUrlForApp;
const navigate = services.application.navigateToApp;
const prompts = services.homeConfig.prompts;
const heroConfig = services.homeConfig.hero;
const isHeroEnabled = heroConfig.enabled;
const prompts = isHeroEnabled ? heroConfig.prompts : [];
type Prompt = typeof prompts extends Array<infer T> ? T : never;

const OLLY_DESCRIPTION = (
<FormattedMessage
id="home.getStarted.ollyDescription"
defaultMessage="Automatically generate complex queries using simple natural language questions. AI assisted summaraies help you navigate and understand your log data."
/>
);

const PLAYGROUND_DESCRIPTION = (
const description = (
<FormattedMessage
id="home.getStarted.playgroundDescription"
defaultMessage="Automatically generate complex queries using simple conversational prompts. AI assisted summary helps you navigate and understand errors from your logs.{br}{br}You will be redirected to the observability playground where you will need to login. All the {terms} of the playground still apply."
defaultMessage={heroConfig.body}
values={{
br: <br />,
terms: (
Expand All @@ -51,45 +47,49 @@ export const GetStartedSection: React.FC<{ olly?: boolean }> = ({ olly = true })
/>
);

const description = olly ? OLLY_DESCRIPTION : PLAYGROUND_DESCRIPTION;

const OLLY_ACTION_BUTTON = (
<EuiButton fullWidth={false} fill href={getUrl('observability-logs', { path: '#/explorer' })}>
<FormattedMessage id="home.getStarted.launchTutorial" defaultMessage="Try in Log Explorer" />
const actionButton = !heroConfig.externalActionButton ? (
<EuiButton
fullWidth={false}
fill
href={getUrl(heroConfig.actionButton.app, { path: heroConfig.actionButton.path })}
>
<FormattedMessage
id="home.getStarted.launchTutorial"
defaultMessage={heroConfig.actionButton.text}
/>
</EuiButton>
);

const PLAYGROUND_ACTION_BUTTON = (
<EuiButton fullWidth={false} fill href="https://observability.playground.opensearch.org/">
<FormattedMessage id="home.getStarted.login" defaultMessage="Login to try" />
) : (
<EuiButton fullWidth={false} fill href={heroConfig.externalActionButton.link}>
<FormattedMessage
id="home.getStarted.login"
defaultMessage={heroConfig.externalActionButton.text}
/>
</EuiButton>
);

const actionButton = olly ? OLLY_ACTION_BUTTON : PLAYGROUND_ACTION_BUTTON;
const content = olly ? (
renderCategories()
) : (
<EuiImage src={illustration} alt="illustration" size="fullWidth" />
);

function renderExample({ prompt, datasourceName, datasourceType, indexPattern }: Prompt) {
function renderExample({ text, app, path }: Prompt) {
return (
<EuiFlexItem key={prompt} grow={false}>
<EuiFlexItem key={text} grow={false}>
<EuiPanel
color="subdued"
paddingSize="s"
onClick={() =>
navigate('observability-logs', {
path: `#/explorer?datasourceName=${encodeURIComponent(
datasourceName
)}&datasourceType=${encodeURIComponent(
datasourceType
)}&indexPattern=${encodeURIComponent(indexPattern)}&olly_q=${encodeURIComponent(
prompt
)}`,
navigate(app, {
path,
})
}
>
<EuiFlexGroup direction="row" responsive={false} alignItems="flexStart" gutterSize="s">
<EuiFlexItem grow={false}>
<EuiIcon type={logo} size="l" />
</EuiFlexItem>
<EuiFlexItem>&quot;{prompt}&quot;</EuiFlexItem>
<EuiFlexItem>&quot;{text}&quot;</EuiFlexItem>
</EuiFlexGroup>
</EuiPanel>
</EuiFlexItem>
Expand All @@ -113,11 +113,14 @@ export const GetStartedSection: React.FC<{ olly?: boolean }> = ({ olly = true })

const links = [
{
text: i18n.translate('home.getStarted.learnMore', { defaultMessage: 'Learn more' }),
url: 'https://opensearch.org/platform/observability/index.html',
text: i18n.translate('home.getStarted.learnMore', {
defaultMessage: heroConfig.secondaryButton.text,
}),
url: heroConfig.secondaryButton.link,
},
];

// TODO: understand why we only want this for Olly
if (olly) {
links.push({
text: i18n.translate('home.getStarted.stayConnected', { defaultMessage: 'Stay connected' }),
Expand All @@ -128,19 +131,30 @@ export const GetStartedSection: React.FC<{ olly?: boolean }> = ({ olly = true })
return (
<HeroSection
title={i18n.translate('home.getStarted.title', {
defaultMessage: 'Try the Query Assistant',
defaultMessage: heroConfig.title,
})}
description={description}
links={links}
actionButton={actionButton}
content={
olly ? (
renderCategories()
content={content}
illustration={
heroConfig.img ? (
<div className="home-hero-illustrationContainer">
<EuiButtonIcon
href={heroConfig.img.link}
aria-labelledby="home-hero-illustrationPlay"
className="home-hero-illustrationButton"
display="fill"
iconType="play"
iconSize="l"
size="m"
/>
<EuiImage src={screenshot} alt="Animated gif 16:9 ratio" />
</div>
) : (
<EuiImage src={illustration} alt="illustration" size="fullWidth" />
<EuiImage src={screenshot} alt="Animated gif 16:9 ratio" />
)
}
illustration={<EuiImage src={image} alt="Animated gif 16:9 ratio" />}
/>
);
};
2 changes: 1 addition & 1 deletion src/plugins/home/server/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ export const config: PluginConfigDescriptor<ConfigSchema> = {
disableWelcomeScreen: true,
disableNewThemeModal: true,
newHomepage: true,
prompts: true,
hero: true,
},
schema: configSchema,
deprecations: ({ renameFromRoot }) => [
Expand Down

0 comments on commit 26cfad2

Please sign in to comment.