Mailspring/app/static/components/tokenizing-text-field.less
Ben Gotow 1a3cca8d0a
Totally overhauled composer based on Slate (#524)
* Remove the composer contenteditable, replace with basic <textarea>

* Beginning broader cleanup of draft session

* DraftJS composer with color, style support

* Serialization/unserialization of basic styles, toolbar working

* WIP

* Switch to draft-js-plugins approach, need to revisit HTML

* Move HTML conversion functionality into plugins

* Add spellcheck context menu to editor

* Initial work on quoted text

* Further work on quoted text

* BLOCK approach

* Entity approach - better, does not bump out to top level

* Hiding and showing quoted text via CSS

* Get rid of ability to inject another subject line component

* Clean up specs, DraftFactory to ES6

* Remove old initial focus hack

* Fix focusing, initial text selection

* Remove participant “collapsing” support, it can be confusing

* Correctly terminate links on carriage returns

* Initial signature support, allow removal of uneditable blocks

* Sync body string with body editorstate

* Simplify draft editor session, finish signatures

* Templates

* Minor fixes

* Simplify link/open tracking, ensure it works

* Reorg composer, rework template editor

* Omg the slowness is all the stupid emoji button

* Polish and small fixes

* Performance improvements, new templates UI

* Don’t assume nodes are elements

* Fix for sending drafts twice due to back-to-back saves

* Fix order of operations on app quit to save drafts reliably

* Improve DraftJS-Convert whitespace handling

* Use contentID throughout attachment lifecycle

* Try to fix images

* Switch to Slate instead of DraftJS… much better

* Fix newline handling

* Bug fixes

* Cleanup

* Finish templates plugin

* Clean up text editing / support for Gmail email styles

* Support for color + size on the same node, clean trailing whitespace

* Restore emoji typeahead / emoji picker

* Fix scrolling in template editor

* Fix specs

* Fix newlines

* Re-implement spellcheck to be faster

* Make spellcheck decorator changes invisible to the undo/redo stack

* Remove comment

* Polish themplates panel

* Fix #521
2018-01-11 15:55:56 -08:00

275 lines
6.7 KiB
Text

@import 'ui-variables';
@token-top: lighten(@background-secondary,0.6%);
@token-bottom: darken(@background-secondary, 2.5%);
@token-hover-top: mix(@token-top, @component-active-color, 92%);
@token-hover-bottom: mix(@token-bottom, @component-active-color, 90%);
@token-selected-top: mix(@token-top, @component-active-color, 15%);
@token-selected-bottom: mix(@token-bottom, @component-active-color, 0%);
@token-invalid-selected-top: mix(@token-top, red, 60%);
@token-invalid-selected-bottom: mix(@token-bottom, red, 55%);
@base-box-shadow: 0 0.5px 0 rgba(0,0,0,0.17),
0 -0.5px 0 rgba(0,0,0,0.17),
0.5px 0 0 rgba(0,0,0,0.17),
-0.5px 0 0 rgba(0,0,0,0.17),
0 1px 1px rgba(0, 0, 0, 0.1);
.tokenizing-field {
display: block;
margin: 0;
padding: 0;
position: relative;
&.disabled {
background: rgba(0, 0, 0, 0.02);
opacity: 0.7;
.token {
background: @token-top;
box-shadow: @base-box-shadow;
&:hover {
background: @token-top;
cursor: default;
box-shadow: @base-box-shadow;
}
}
.tokenizing-field-input {
&:hover {
cursor: default;
}
}
&:hover {
cursor: default;
}
}
.content-container {
border: 1px solid @border-color-secondary;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
border-radius: @border-radius-small;
background-color: @background-primary;
position: absolute;
z-index: 3;
}
.content-container.empty {
border: 0;
box-shadow: none;
}
.header-container,
.footer-container {
background-color: transparent;
padding: 0;
margin: 0;
border: 0;
}
.token-editing-input {
max-width: 100%;
font-size: 15px;
line-height: 17px;
padding: 2em; //0.5em @spacing-three-quarters 0.5em @spacing-three-quarters;
padding-right: 1.5em;
margin: 3px 6px 6px 1px;
}
.token {
display: inline-block;
position: relative;
color: @text-color;
padding: 0 @spacing-three-quarters;
padding-right: 1.5em;
margin: 1px 5px 1px 1px;
border-radius: @border-radius-base * 0.8;
max-width: 100%;
line-height: 1.9em;
background: linear-gradient(to bottom, @token-top 0%, @token-bottom 100%);
box-shadow: @base-box-shadow;
vertical-align: middle;
.action {
position: absolute;
padding: 0;
border: 0;
margin: 0;
right: 7px;
background-color: transparent;
color: @text-color-very-subtle;
img {
background-color: @text-color-very-subtle;
}
font-size: 10px;
}
&:hover {
background: linear-gradient(to bottom, @token-hover-top 0%, @token-hover-bottom 100%);
box-shadow: 0 0.5px 0 darken(@token-hover-bottom, 35%),
0 -0.5px 0 darken(@token-hover-top, 25%), 0.5px 0 0 darken(@token-hover-bottom, 25%),
-0.5px 0 0 darken(@token-hover-bottom, 25%), 0 1px 1px rgba(0, 0, 0, 0.07);
cursor: default;
}
&.invalid {
border-bottom: 1px dashed red;
margin-bottom: -1px;
background: transparent;
&:hover {
box-shadow: 0 -0.5px 0 @token-invalid-selected-top, 0.5px 0 0 @token-invalid-selected-bottom,
-0.5px 0 0 @token-invalid-selected-bottom, 0 1px 1px rgba(0, 0, 0, 0.07);
}
}
&.invalid.selected,
&.invalid.dragging {
background: linear-gradient(
to bottom,
@token-invalid-selected-top 0%,
@token-invalid-selected-bottom 100%
);
box-shadow: inset 0 1.5px 0 rgba(255, 255, 255, 0.3), 0 1px 1px rgba(0, 0, 0, 0.1);
border: 1px solid darken(@token-invalid-selected-bottom, 8%);
border-top: 1px solid darken(@token-invalid-selected-top, 10%);
}
&.selected,
&.dragging {
background: linear-gradient(to bottom, @token-selected-top 0%, @token-selected-bottom 100%);
box-shadow: inset 0 1.5px 0 rgba(255, 255, 255, 0.3), 0 1px 1px rgba(0, 0, 0, 0.1);
border: 1px solid darken(@token-selected-bottom, 8%);
border-top: 1px solid darken(@token-selected-top, 10%);
border-radius: @border-radius-base;
// Note: we switch from 0.5px borders with box shadows to a real border,
// because the 0.5px shadows can't be as dark as we want. This means
// margins / border radius change by 1px.
margin: 0 4px 0 0;
color: @text-color-inverse;
.action {
color: @text-color-inverse-subtle;
img {
background-color: @text-color-inverse-subtle;
}
}
.secondary,
.participant-secondary {
color: @text-color-inverse-subtle;
}
}
&.dragging {
cursor: -webkit-grabbing;
}
}
.tokenizing-field-label {
color: @text-color-very-subtle;
float: left;
text-transform: capitalize;
padding-top: 12px;
display: block;
&:hover {
cursor: default;
}
}
.tokenizing-field-input {
position: relative;
margin-left: 2.8em;
padding-top: 9px;
padding-bottom: 3px;
&:hover {
cursor: text;
}
&.at-max-tokens {
cursor: default;
input {
&:hover {
cursor: default;
}
opacity: 0;
}
}
input {
display: inline-block;
width: initial;
vertical-align: top;
border: none;
min-width: 0.5em;
background-color: transparent;
// NOTE: padding-top and padding-bottom need to match that of
// `.token`. to ensure they have the same baseline.
//
// The padding-left and padding-right must be 0 so we can manually
// set the width properly to always match the size of the input
// test.
padding: 0.3em 0 0.3em 0;
&.noop-input {
position: absolute;
min-width: 0;
padding-left: 0;
margin-right: 0;
box-shadow: none;
&:focus {
box-shadow: none;
}
}
}
input:focus {
box-shadow: none;
}
}
.placeholder {
color: fade(@text-color, 50%);
position: absolute;
top: 50%;
margin-top: -0.75em;
left: 8px;
}
}
body.is-blurred .tokenizing-field .token:not(.invalid) {
background: @background-secondary;
color: @text-color;
&.selected .action img {
background-color: @text-color-very-subtle;
}
}
body.platform-win32 {
.tokenizing-field {
.content-container {
border-radius: 0;
}
.tokenizing-field-input {
input,
input:focus {
border: 0;
box-shadow: 0 0 0;
}
}
}
.token {
border-radius: 0;
background: @token-bottom;
&.selected,
&.dragging {
border-radius: 0;
background: @token-selected-top;
}
&:hover {
background: @token-hover-bottom;
}
&.invalid.selected,
&.invalid.dragging {
background: @token-invalid-selected-top;
}
}
}