Skip to content

Commit

Permalink
Better target sequence selection for LM salience module
Browse files Browse the repository at this point in the history
- Show an interstitial with target selection when selecting a new example.
- Pending request indicators when waiting for model generation.
- Possible now to inspect salience on 'target' while waiting for model generation.
- Better help text in the dropdown selector.
- Better behavior when the reference ("target") is empty.

PiperOrigin-RevId: 607841998
  • Loading branch information
iftenney authored and LIT team committed Feb 17, 2024
1 parent ca032ff commit 71e6800
Show file tree
Hide file tree
Showing 3 changed files with 190 additions and 40 deletions.
4 changes: 3 additions & 1 deletion lit_nlp/client/core/lit_module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -87,8 +87,10 @@ export abstract class LitModule extends ReactiveElement {
@observable @property({type: String}) model = '';
@observable @property({type: Number}) selectionServiceIndex = 0;

// tslint:disable-next-line:no-any
// tslint:disable:no-any
@observable
protected readonly latestLoadPromises = new Map<string, Promise<any>>();
// tslint:enable:no-any

protected readonly apiService = app.getService(ApiService);
protected readonly appState = app.getService(AppState);
Expand Down
83 changes: 77 additions & 6 deletions lit_nlp/client/modules/lm_salience_module.css
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,72 @@ lit-switch .icon-button {
vertical-align: middle;
}

select:invalid {
color: var(--lit-neutral-400);
}

/**
* Interstitial for target selection
*/
.interstitial-container {
height: 100%;
width: 100%;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}

.interstitial-header {
font-size: 1.5em;
font-family: 'Google Sans', sans;
padding: 4px 2px; /* 2px left shift vs. smaller fonts below */
}

.interstitial-subtitle {
font-family: 'Google Sans', sans;
color: var(--lit-neutral-500);
padding: 4px;
margin-bottom: 8px;
}

.interstitial-contents {
width: 80%;
/* color: #5f6368; */
color: var(--lit-neutral-700);
}

.interstitial-target-option {
display: flex;
flex-direction: row;
margin-bottom: 6px;
margin-left: 4px;
cursor: pointer;
}

.interstitial-target-type {
/* width: 80px; */
/* font-weight: bold; */
color: var(--lit-neutral-500);
padding: 4px;
}

.interstitial-target-option .interstitial-target-text {
flex-grow: 1;
flex-basis: 100%;
padding: 4px;
border: 1px solid var(--lit-neutral-300);
border-radius: 4px;
overflow: hidden;
min-height: 28px; /* one line of text */
max-height: 34px; /* two lines of text */
}

.interstitial-target-option:hover .interstitial-target-text {
border: 1px solid var(--lit-neutral-400);
background-color: var(--lit-mintonal-p-1);
}

/**
* Module controls
*/
Expand Down Expand Up @@ -60,9 +126,13 @@ lit-switch .icon-button {
margin-right: 8px;
}

.controls-group-variable .dropdown {
max-width: calc(100% - 22px);
.target-dropdown-holder {
width: calc(100% - 22px);
margin-right: 4px;
}

.target-dropdown-holder .dropdown {
width: 100%;
text-overflow: ellipsis;
}

Expand Down Expand Up @@ -97,17 +167,18 @@ color-legend {
.loading-indicator-container {
position: relative;
width: 100%;
top: -2px;
}


@keyframes running-progress {
0% { margin-left: 0; margin-right: 100%; }
50% { margin-left: 35%; margin-right: 0%; }
100% { margin-left: 100%; margin-right: 0%; }
0% { margin-left: 0; width: 0; }
50% { margin-left: 35%; width: 65%; }
100% { margin-left: 100%; width: 0; }
}

.loading-indicator {
position: absolute;
top: -2px;
background-color: var(--lit-neutral-500);
width: 100%;
height: 2px;
Expand Down
143 changes: 110 additions & 33 deletions lit_nlp/client/modules/lm_salience_module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -211,7 +211,7 @@ export class LMSalienceModule extends SingleExampleSingleModelModule {

@observable.ref private currentTokens: string[] = [];
@observable.ref private salienceTargetOptions: TargetOption[] = [];
@observable private salienceTargetString = '';
@observable private salienceTargetOption?: number; // index into above
@observable.ref private targetSegmentSpan?: [number, number] = undefined;


Expand Down Expand Up @@ -254,37 +254,23 @@ export class LMSalienceModule extends SingleExampleSingleModelModule {
// Generation & target string selection
super.resetState(); // currentData and currentPreds
this.salienceTargetOptions = [];
this.salienceTargetString = '';
this.salienceTargetOption = undefined;
// Tokens and selected target span
this.currentTokens = [];
this.resetTargetSpan();
// Salience results
this.salienceResultCache = {};
}

// Get generations; populate this.currentPreds
protected override async updateToSelection() {
await super.updateToSelection();
this.resetTargetSpan();

const dataSpec = this.appState.currentDatasetSpec;
const outputSpec = this.appState.getModelSpec(this.model).output;
this.salienceTargetOptions = getAllTargetOptions(
dataSpec,
outputSpec,
this.currentData,
this.currentPreds,
);
this.salienceTargetString = this.salienceTargetOptions[0]?.text ?? '';
}

// Modified input with selected target sequence. Use this for tokens and
// salience.
@computed
get modifiedData(): IndexedInput|null {
if (this.currentData == null) return null;
return makeModifiedInput(
this.currentData, {'target': this.salienceTargetString});
if (this.salienceTargetOption === undefined) return null;
const targetString =
this.salienceTargetOptions[this.salienceTargetOption].text;
return makeModifiedInput(this.currentData, {'target': targetString});
}

@computed
Expand Down Expand Up @@ -496,6 +482,20 @@ export class LMSalienceModule extends SingleExampleSingleModelModule {
override firstUpdated() {
super.firstUpdated();

// Update target options based on current data and preds.
// TODO: could this just be @computed?
// If we maintain explicit state, we can support custom target strings.
this.reactImmediately(() => [this.currentData, this.currentPreds], () => {
const dataSpec = this.appState.currentDatasetSpec;
const outputSpec = this.appState.getModelSpec(this.model).output;
this.salienceTargetOptions = getAllTargetOptions(
dataSpec,
outputSpec,
this.currentData,
this.currentPreds,
);
});

// If selected example OR selected target string change.
// NOTE: you may see a console warning: "Element lm-salience-module
// scheduled an update (generally because a property was set) after an
Expand Down Expand Up @@ -621,38 +621,53 @@ export class LMSalienceModule extends SingleExampleSingleModelModule {

renderSalienceTargetStringSelector() {
const onChangeTarget = (e: Event) => {
this.salienceTargetString = (e.target as HTMLInputElement).value;
const value = (e.target as HTMLInputElement).value;
this.salienceTargetOption = value !== '' ? +value : undefined;
};

const options = this.salienceTargetOptions.map(target => {
const targetSelectorHelp =
'Select a (response) from the model or a pre-defined (target) sequence from the dataset.';

const options = this.salienceTargetOptions.map((target, i) => {
// TODO(b/324959547): get field names 'target' and 'response' from spec
// via generated_text_utils.ts, rather than hard-coding.
// This information is available on the frontend, but we need to thread
// it through a few layers of code in generated_text_utils.ts
const sourceName =
target.source === TargetSource.REFERENCE ? 'target' : 'response';
return html`<option value=${target.text}
?selected=${target.text === this.salienceTargetString}>
// prettier-ignore
return html`
<option value=${i} ?selected=${i === this.salienceTargetOption}>
(${sourceName}) "${target.text}"
</option>`;
</option>`;
});
// Empty default option. Styled as select:invalid.
// prettier-ignore
options.unshift(html`
<option value='' disabled hidden ?selected=${
this.salienceTargetOption === undefined}>
${targetSelectorHelp}
</option>`);

const targetSelectorHelp =
'Select a (response) from the model or a pre-defined (target) sequence from the dataset.';
const isLoadingPreds = this.latestLoadPromises.has('modelPreds');

// prettier-ignore
return html`
<div class="controls-group controls-group-variable"
title="Target string for salience.">
<select class="dropdown" @change=${onChangeTarget}>
${options}
</select>
title=${targetSelectorHelp}>
<div class='target-dropdown-holder'>
<select required class='dropdown' @change=${onChangeTarget}>
${options}
</select>
${isLoadingPreds ? this.renderLoadingIndicator() : null}
</div>
<lit-tooltip content=${targetSelectorHelp} tooltipPosition="left">
<span class="help-icon material-icon-outlined icon-button">
help_outline
</span>
</lit-tooltip>
</div>`;
</div>
`;
}

renderLoadingIndicator() {
Expand Down Expand Up @@ -681,7 +696,6 @@ export class LMSalienceModule extends SingleExampleSingleModelModule {
const requestPending = this.targetTokenSpan !== undefined &&
this.salienceResultCache[this.spanToKey(this.targetTokenSpan)] ===
REQUEST_PENDING;
// const requestPending = true;
const infoLineClasses = classMap({
'target-info-line': true,
'gray-text': requestPending,
Expand Down Expand Up @@ -744,7 +758,70 @@ export class LMSalienceModule extends SingleExampleSingleModelModule {
return i >= this.targetSegmentSpan[0] && i < this.targetSegmentSpan[1];
}

renderTargetSelectorInterstitial() {
const formatOption = (target: TargetOption, i: number) => {
const onClickTarget = () => {
this.salienceTargetOption = i;
};
// prettier-ignore
return html`
<div class='interstitial-target-option' @click=${onClickTarget}>
<div class='interstitial-target-text'>${target.text}</div>
</div>`;
};

// Slightly awkward, but we need to process and /then/ filter, because
// the @click handler needs the original list index.
const optionsFromDataset =
this.salienceTargetOptions
.map((target, i) => {
if (target.source !== TargetSource.REFERENCE) return null;
return formatOption(target, i);
})
.filter(val => val != null);
const optionsFromModel =
this.salienceTargetOptions
.map((target, i) => {
if (target.source !== TargetSource.MODEL_OUTPUT) return null;
return formatOption(target, i);
})
.filter(val => val != null);

const isLoadingPreds = this.latestLoadPromises.has('modelPreds');

// TODO(b/324959547): get field names 'target' and 'response' from spec
// via generated_text_utils.ts, rather than hard-coding.
// This information is available on the frontend, but we need to thread
// it through a few layers of code in generated_text_utils.ts

// prettier-ignore
return html`
<div class='interstitial-container'>
<div class='interstitial-contents'>
<div class='interstitial-header'>
Choose a sequence to explain
</div>
<div class='interstitial-subtitle'>
Or select from the dropdown at the top of this module
</div>
<div class='interstitial-target-selector'>
<div class='interstitial-target-type'>From dataset (target):</div>
${optionsFromDataset}
<div class='interstitial-target-type'>From model (response):</div>
${isLoadingPreds ? this.renderLoadingIndicator() : null}
${optionsFromModel}
</div>
</div>
</div>`;
}

renderContent() {
if (this.currentData == null) return null;

if (this.salienceTargetOption === undefined) {
return this.renderTargetSelectorInterstitial();
}

if (this.currentSegmentTexts.length === 0) return null;

const segments: string[] = this.currentSegmentTexts;
Expand Down

0 comments on commit 71e6800

Please sign in to comment.