mirror of
https://github.com/elastic/kibana.git
synced 2025-04-23 17:28:26 -04:00
* First changes for avatar images * Added the ability to have custom images for space avatars * Partial changes as requested by reviewers * Final commit for space avatar images PR * Wrapping avatar file name * Colour picker always enabled, to allow background change for transparent svgs * All the changes requested in the last review * Fixes the type_check test errors * Fixing the rendering errors for space pages * Another batch of changes as requested by review * Some more snapshot tests * Last batch of changes * Fixed the type_check test * API doc updates * Removed comment * Removed imageUrl from state Co-authored-by: Larry Gregory <larry.gregory@elastic.co> Co-authored-by: Elastic Machine <elasticmachine@users.noreply.github.com>
This commit is contained in:
parent
927204cda9
commit
45cc8d2aa5
17 changed files with 295 additions and 6 deletions
|
@ -32,6 +32,7 @@ The API returns the following:
|
|||
"description" : "This is the Marketing Space",
|
||||
"color": "#aabbcc",
|
||||
"initials": "MK",
|
||||
"disabledFeatures": []
|
||||
"disabledFeatures": [],
|
||||
"imageUrl": ""
|
||||
}
|
||||
--------------------------------------------------
|
|
@ -32,6 +32,7 @@ The API returns the following:
|
|||
"name": "Default",
|
||||
"description" : "This is the Default Space",
|
||||
"disabledFeatures": [],
|
||||
"imageUrl": "",
|
||||
"_reserved": true
|
||||
},
|
||||
{
|
||||
|
@ -40,13 +41,15 @@ The API returns the following:
|
|||
"description" : "This is the Marketing Space",
|
||||
"color": "#aabbcc",
|
||||
"disabledFeatures": ["apm"],
|
||||
"initials": "MK"
|
||||
"initials": "MK",
|
||||
"imageUrl": "",
|
||||
},
|
||||
{
|
||||
"id": "sales",
|
||||
"name": "Sales",
|
||||
"initials": "MK",
|
||||
"disabledFeatures": ["discover", "timelion"],
|
||||
"imageUrl": ""
|
||||
},
|
||||
]
|
||||
--------------------------------------------------
|
||||
|
|
|
@ -33,6 +33,10 @@ experimental["The underlying Spaces concepts are stable, but the APIs for managi
|
|||
|
||||
`color`::
|
||||
(Optional, string) Specifies the hexadecimal color code used in the space avatar. By default, the color is automatically generated from the space name.
|
||||
|
||||
`imageUrl`::
|
||||
(Optional, string) Specifies the data-url encoded image to display in the space avatar. If specified, `initials` will not be displayed, and the `color` will be visible as the background color for transparent images.
|
||||
For best results, your image should be 64x64. Images will not be optimized by this API call, so care should be taken when using custom images.
|
||||
|
||||
[[spaces-api-post-response-codes]]
|
||||
==== Response codes
|
||||
|
@ -52,7 +56,8 @@ POST /api/spaces/space
|
|||
"description" : "This is the Marketing Space",
|
||||
"color": "#aabbcc",
|
||||
"initials": "MK",
|
||||
"disabledFeatures": ["timelion"]
|
||||
"disabledFeatures": ["timelion"],
|
||||
"imageUrl": ""
|
||||
}
|
||||
--------------------------------------------------
|
||||
// KIBANA
|
||||
|
|
|
@ -34,6 +34,10 @@ experimental["The underlying Spaces concepts are stable, but the APIs for managi
|
|||
`color`::
|
||||
(Optional, string) Specifies the hexadecimal color code used in the space avatar. By default, the color is automatically generated from the space name.
|
||||
|
||||
`imageUrl`::
|
||||
(Optional, string) Specifies the data-url encoded image to display in the space avatar. If specified, `initials` will not be displayed, and the `color` will be visible as the background color for transparent images.
|
||||
For best results, your image should be 64x64. Images will not be optimized by this API call, so care should be taken when using custom images.
|
||||
|
||||
[[spaces-api-put-response-codes]]
|
||||
==== Response codes
|
||||
|
||||
|
@ -52,7 +56,8 @@ PUT /api/spaces/space/marketing
|
|||
"description" : "This is the Marketing Space",
|
||||
"color": "#aabbcc",
|
||||
"initials": "MK",
|
||||
"disabledFeatures": []
|
||||
"disabledFeatures": [],
|
||||
"imageUrl": ""
|
||||
}
|
||||
--------------------------------------------------
|
||||
// KIBANA
|
||||
|
|
25
x-pack/legacy/plugins/spaces/common/lib/dataurl.ts
Normal file
25
x-pack/legacy/plugins/spaces/common/lib/dataurl.ts
Normal file
|
@ -0,0 +1,25 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
|
||||
import { fromByteArray } from 'base64-js';
|
||||
|
||||
export const imageTypes = ['image/svg+xml', 'image/jpeg', 'image/png', 'image/gif'];
|
||||
|
||||
export function encode(data: any | null, type = 'text/plain') {
|
||||
// use FileReader if it's available, like in the browser
|
||||
if (FileReader) {
|
||||
return new Promise<string>((resolve, reject) => {
|
||||
const reader = new FileReader();
|
||||
reader.onloadend = () => resolve(reader.result as string);
|
||||
reader.onerror = err => reject(err);
|
||||
reader.readAsDataURL(data);
|
||||
});
|
||||
}
|
||||
|
||||
// otherwise fall back to fromByteArray
|
||||
// note: Buffer doesn't seem to correctly base64 encode binary data
|
||||
return Promise.resolve(`data:${type};base64,${fromByteArray(data)}`);
|
||||
}
|
|
@ -12,4 +12,5 @@ export interface Space {
|
|||
initials?: string;
|
||||
disabledFeatures: string[];
|
||||
_reserved?: boolean;
|
||||
imageUrl?: string;
|
||||
}
|
||||
|
|
|
@ -52,3 +52,18 @@ export function getSpaceInitials(space: Partial<Space> = {}) {
|
|||
|
||||
return words.map(word => word.substring(0, 1)).join('');
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines the avatar image for the provided space.
|
||||
*
|
||||
* @param {Space} space
|
||||
*/
|
||||
export function getSpaceImageUrl(space: Partial<Space> = {}) {
|
||||
const { imageUrl } = space;
|
||||
|
||||
if (imageUrl) {
|
||||
return imageUrl;
|
||||
}
|
||||
|
||||
return '';
|
||||
}
|
||||
|
|
|
@ -22,6 +22,10 @@
|
|||
"disabledFeatures": {
|
||||
"type": "keyword"
|
||||
},
|
||||
"imageUrl": {
|
||||
"type": "text",
|
||||
"index": false
|
||||
},
|
||||
"_reserved": {
|
||||
"type": "boolean"
|
||||
}
|
||||
|
|
|
@ -15,6 +15,7 @@ exports[`removes aria-label when instructed not to announce the space name 1`] =
|
|||
aria-label=""
|
||||
color="#BFA180"
|
||||
data-test-subj="space-avatar-"
|
||||
imageUrl=""
|
||||
initials=""
|
||||
initialsLength={2}
|
||||
name=""
|
||||
|
@ -43,6 +44,7 @@ exports[`renders with a space name entirely made of whitespace 1`] = `
|
|||
<EuiAvatar
|
||||
color="#DB1374"
|
||||
data-test-subj="space-avatar-"
|
||||
imageUrl=""
|
||||
initials=""
|
||||
initialsLength={2}
|
||||
name=""
|
||||
|
@ -55,6 +57,7 @@ exports[`renders without crashing 1`] = `
|
|||
<EuiAvatar
|
||||
color="#BFA180"
|
||||
data-test-subj="space-avatar-"
|
||||
imageUrl=""
|
||||
initials=""
|
||||
initialsLength={2}
|
||||
name=""
|
||||
|
|
|
@ -8,6 +8,7 @@ import { EuiAvatar, isValidHex } from '@elastic/eui';
|
|||
import React, { SFC } from 'react';
|
||||
import { getSpaceColor, getSpaceInitials, MAX_SPACE_INITIALS } from '../../common';
|
||||
import { Space } from '../../common/model/space';
|
||||
import { getSpaceImageUrl } from '../../common/space_attributes';
|
||||
|
||||
interface Props {
|
||||
space: Partial<Space>;
|
||||
|
@ -37,6 +38,7 @@ export const SpaceAvatar: SFC<Props> = (props: Props) => {
|
|||
initialsLength={MAX_SPACE_INITIALS}
|
||||
initials={getSpaceInitials(space)}
|
||||
color={isValidHex(spaceColor) ? spaceColor : ''}
|
||||
imageUrl={getSpaceImageUrl(space)}
|
||||
{...rest}
|
||||
/>
|
||||
);
|
||||
|
|
|
@ -14,6 +14,7 @@ exports[`renders without crashing 1`] = `
|
|||
>
|
||||
<EuiFieldText
|
||||
compressed={false}
|
||||
disabled={false}
|
||||
fullWidth={false}
|
||||
inputRef={[Function]}
|
||||
isLoading={false}
|
||||
|
@ -40,5 +41,31 @@ exports[`renders without crashing 1`] = `
|
|||
onChange={[Function]}
|
||||
/>
|
||||
</EuiFormRow>
|
||||
<EuiSpacer
|
||||
size="m"
|
||||
/>
|
||||
<EuiFormRow
|
||||
describedByIds={Array []}
|
||||
display="row"
|
||||
fullWidth={false}
|
||||
hasEmptyLabelSpace={false}
|
||||
label="Custom image"
|
||||
labelType="label"
|
||||
>
|
||||
<EuiFilePicker
|
||||
accept={
|
||||
Array [
|
||||
"image/svg+xml",
|
||||
"image/jpeg",
|
||||
"image/png",
|
||||
"image/gif",
|
||||
]
|
||||
}
|
||||
compressed={false}
|
||||
display="default"
|
||||
initialPromptText="Select image file"
|
||||
onChange={[Function]}
|
||||
/>
|
||||
</EuiFormRow>
|
||||
</form>
|
||||
`;
|
||||
|
|
|
@ -4,9 +4,22 @@
|
|||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
|
||||
import { EuiColorPicker, EuiFieldText, EuiSpacer, EuiFormRow, isValidHex } from '@elastic/eui';
|
||||
import { InjectedIntl, injectI18n } from '@kbn/i18n/react';
|
||||
import React, { ChangeEvent, Component } from 'react';
|
||||
import {
|
||||
EuiColorPicker,
|
||||
EuiFieldText,
|
||||
EuiFlexItem,
|
||||
EuiFormRow,
|
||||
// @ts-ignore (elastic/eui#1262) EuiFilePicker is not exported yet
|
||||
EuiFilePicker,
|
||||
EuiButton,
|
||||
EuiSpacer,
|
||||
isValidHex,
|
||||
} from '@elastic/eui';
|
||||
import { InjectedIntl, injectI18n } from '@kbn/i18n/react';
|
||||
|
||||
import { encode, imageTypes } from '../../../../../common/lib/dataurl';
|
||||
|
||||
import { MAX_SPACE_INITIALS } from '../../../../../common/constants';
|
||||
import { Space } from '../../../../../common/model/space';
|
||||
import { getSpaceColor, getSpaceInitials } from '../../../../../common/space_attributes';
|
||||
|
@ -32,6 +45,63 @@ class CustomizeSpaceAvatarUI extends Component<Props, State> {
|
|||
};
|
||||
}
|
||||
|
||||
private storeImageChanges(imageUrl: string) {
|
||||
this.props.onChange({
|
||||
...this.props.space,
|
||||
imageUrl,
|
||||
});
|
||||
}
|
||||
|
||||
//
|
||||
// images below 64x64 pixels are left untouched
|
||||
// images above that threshold are resized
|
||||
//
|
||||
|
||||
private handleImageUpload = (imgUrl: string) => {
|
||||
const thisInstance = this;
|
||||
const image = new Image();
|
||||
image.addEventListener(
|
||||
'load',
|
||||
function() {
|
||||
const MAX_IMAGE_SIZE = 64;
|
||||
const imgDimx = image.width;
|
||||
const imgDimy = image.height;
|
||||
if (imgDimx <= MAX_IMAGE_SIZE && imgDimy <= MAX_IMAGE_SIZE) {
|
||||
thisInstance.storeImageChanges(imgUrl);
|
||||
} else {
|
||||
const imageCanvas = document.createElement('canvas');
|
||||
const canvasContext = imageCanvas.getContext('2d');
|
||||
if (imgDimx >= imgDimy) {
|
||||
imageCanvas.width = MAX_IMAGE_SIZE;
|
||||
imageCanvas.height = Math.floor((imgDimy * MAX_IMAGE_SIZE) / imgDimx);
|
||||
if (canvasContext) {
|
||||
canvasContext.drawImage(image, 0, 0, imageCanvas.width, imageCanvas.height);
|
||||
const resizedImageUrl = imageCanvas.toDataURL();
|
||||
thisInstance.storeImageChanges(resizedImageUrl);
|
||||
}
|
||||
} else {
|
||||
imageCanvas.height = MAX_IMAGE_SIZE;
|
||||
imageCanvas.width = Math.floor((imgDimx * MAX_IMAGE_SIZE) / imgDimy);
|
||||
if (canvasContext) {
|
||||
canvasContext.drawImage(image, 0, 0, imageCanvas.width, imageCanvas.height);
|
||||
const resizedImageUrl = imageCanvas.toDataURL();
|
||||
thisInstance.storeImageChanges(resizedImageUrl);
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
false
|
||||
);
|
||||
image.src = imgUrl;
|
||||
};
|
||||
|
||||
private onFileUpload = (files: File[]) => {
|
||||
const [file] = files;
|
||||
if (imageTypes.indexOf(file.type) > -1) {
|
||||
encode(file).then((dataurl: string) => this.handleImageUpload(dataurl));
|
||||
}
|
||||
};
|
||||
|
||||
public render() {
|
||||
const { space, intl } = this.props;
|
||||
|
||||
|
@ -55,6 +125,7 @@ class CustomizeSpaceAvatarUI extends Component<Props, State> {
|
|||
// without defaulting to the derived initials provided by `getSpaceInitials`
|
||||
value={initialsHasFocus ? pendingInitials || '' : getSpaceInitials(space)}
|
||||
onChange={this.onInitialsChange}
|
||||
disabled={this.props.space.imageUrl && this.props.space.imageUrl !== '' ? true : false}
|
||||
/>
|
||||
</EuiFormRow>
|
||||
<EuiSpacer size="m" />
|
||||
|
@ -71,10 +142,55 @@ class CustomizeSpaceAvatarUI extends Component<Props, State> {
|
|||
isInvalid={isInvalidSpaceColor}
|
||||
/>
|
||||
</EuiFormRow>
|
||||
<EuiSpacer size="m" />
|
||||
{this.filePickerOrImage()}
|
||||
</form>
|
||||
);
|
||||
}
|
||||
|
||||
private removeImageUrl() {
|
||||
this.props.onChange({
|
||||
...this.props.space,
|
||||
imageUrl: '',
|
||||
});
|
||||
}
|
||||
|
||||
public filePickerOrImage() {
|
||||
const { intl } = this.props;
|
||||
|
||||
if (!this.props.space.imageUrl) {
|
||||
return (
|
||||
<EuiFormRow
|
||||
label={intl.formatMessage({
|
||||
id: 'xpack.spaces.management.customizeSpaceAvatar.imageUrl',
|
||||
defaultMessage: 'Custom image',
|
||||
})}
|
||||
>
|
||||
<EuiFilePicker
|
||||
display="default"
|
||||
initialPromptText={intl.formatMessage({
|
||||
id: 'xpack.spaces.management.customizeSpaceAvatar.selectImageUrl',
|
||||
defaultMessage: 'Select image file',
|
||||
})}
|
||||
onChange={this.onFileUpload}
|
||||
accept={imageTypes}
|
||||
/>
|
||||
</EuiFormRow>
|
||||
);
|
||||
} else {
|
||||
return (
|
||||
<EuiFlexItem grow={true}>
|
||||
<EuiButton onClick={() => this.removeImageUrl()} color="danger" iconType="trash">
|
||||
{intl.formatMessage({
|
||||
id: 'xpack.spaces.management.customizeSpaceAvatar.removeImage',
|
||||
defaultMessage: 'Remove custom image',
|
||||
})}
|
||||
</EuiButton>
|
||||
</EuiFlexItem>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
public initialsInputRef = (ref: HTMLInputElement) => {
|
||||
if (ref) {
|
||||
this.initialsRef = ref;
|
||||
|
|
|
@ -334,6 +334,7 @@ class ManageSpacePageUI extends Component<Props, State> {
|
|||
initials,
|
||||
color,
|
||||
disabledFeatures = [],
|
||||
imageUrl,
|
||||
} = this.state.space;
|
||||
|
||||
const params = {
|
||||
|
@ -343,6 +344,7 @@ class ManageSpacePageUI extends Component<Props, State> {
|
|||
initials,
|
||||
color,
|
||||
disabledFeatures,
|
||||
imageUrl,
|
||||
};
|
||||
|
||||
let action;
|
||||
|
|
|
@ -145,6 +145,7 @@ Array [
|
|||
<EuiAvatar
|
||||
color="#461A0A"
|
||||
data-test-subj="space-avatar-default"
|
||||
imageUrl=""
|
||||
initials="D"
|
||||
initialsLength={2}
|
||||
name="Default"
|
||||
|
@ -186,6 +187,7 @@ Array [
|
|||
<EuiAvatar
|
||||
color="#BFA180"
|
||||
data-test-subj="space-avatar-custom-1"
|
||||
imageUrl=""
|
||||
initials="C1"
|
||||
initialsLength={2}
|
||||
name="Custom 1"
|
||||
|
@ -230,6 +232,7 @@ Array [
|
|||
<EuiAvatar
|
||||
color="#ABCDEF"
|
||||
data-test-subj="space-avatar-custom-2"
|
||||
imageUrl=""
|
||||
initials="LG"
|
||||
initialsLength={2}
|
||||
name="Custom 2"
|
||||
|
|
|
@ -136,3 +136,32 @@ describe('#color', () => {
|
|||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe('#imageUrl', () => {
|
||||
test('is optional', () => {
|
||||
const result = spaceSchema.validate({
|
||||
...defaultProperties,
|
||||
imageUrl: undefined,
|
||||
});
|
||||
expect(result.error).toBeNull();
|
||||
});
|
||||
|
||||
test(`must start with data:image`, () => {
|
||||
const result = spaceSchema.validate({
|
||||
...defaultProperties,
|
||||
imageUrl: 'notValid',
|
||||
});
|
||||
expect(result.error).toMatchInlineSnapshot(
|
||||
`[ValidationError: child "imageUrl" fails because ["imageUrl" with value "notValid" fails to match the Image URL should start with 'data:image' pattern]]`
|
||||
);
|
||||
});
|
||||
|
||||
test(`checking that a valid image is accepted as imageUrl`, () => {
|
||||
const result = spaceSchema.validate({
|
||||
...defaultProperties,
|
||||
imageUrl:
|
||||
'',
|
||||
});
|
||||
expect(result.error).toBeNull();
|
||||
});
|
||||
});
|
||||
|
|
|
@ -19,4 +19,7 @@ export const spaceSchema = Joi.object({
|
|||
.items(Joi.string())
|
||||
.default([]),
|
||||
_reserved: Joi.boolean(),
|
||||
imageUrl: Joi.string()
|
||||
.allow('')
|
||||
.regex(/^data:image.*$/, `Image URL should start with 'data:image'`),
|
||||
}).default();
|
||||
|
|
|
@ -27,5 +27,50 @@ export default function({ getService }: FtrProviderContext) {
|
|||
color: '#aaBB78',
|
||||
});
|
||||
});
|
||||
|
||||
it('should allow a space to be created with an avatar image', async () => {
|
||||
await supertest
|
||||
.post('/api/spaces/space')
|
||||
.set('kbn-xsrf', 'xxx')
|
||||
.send({
|
||||
id: 'api-test-space2',
|
||||
name: 'Space with image',
|
||||
disabledFeatures: [],
|
||||
color: '#cafeba',
|
||||
imageUrl:
|
||||
'',
|
||||
})
|
||||
.expect(200, {
|
||||
id: 'api-test-space2',
|
||||
name: 'Space with image',
|
||||
disabledFeatures: [],
|
||||
color: '#cafeba',
|
||||
imageUrl:
|
||||
'',
|
||||
});
|
||||
});
|
||||
|
||||
it('creating a space with an invalid image fails', async () => {
|
||||
await supertest
|
||||
.post('/api/spaces/space')
|
||||
.set('kbn-xsrf', 'xxx')
|
||||
.send({
|
||||
id: 'api-test-space3',
|
||||
name: 'Space with invalid image',
|
||||
disabledFeatures: [],
|
||||
color: '#cafeba',
|
||||
imageUrl: 'invalidImage',
|
||||
})
|
||||
.expect(400, {
|
||||
error: 'Bad Request',
|
||||
message:
|
||||
'child "imageUrl" fails because ["imageUrl" with value "invalidImage" fails to match the Image URL should start with \'data:image\' pattern]',
|
||||
statusCode: 400,
|
||||
validation: {
|
||||
keys: ['imageUrl'],
|
||||
source: 'payload',
|
||||
},
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue