The Secrets of Keeping Your Content Editors Happy
Our client is migrating from Luminate CMS to Drupal because they want to improve performance without changing the look or feel of the site. Each of the pages on a Luminate site are like snowflakes - unique. It doesn’t make sense to rebuild those features as structured blocks given that they only appear on one single page. So having the ability to use existing JS and CSS allows us to copy and paste markup without rebuilding a whole structure that wouldn’t be repurposed on other pages.
This technically savvy client wants a way to add existing JavaScript and CSS to Drupal pages. So let’s give them the capability of putting raw CSS and JavaScript on their pages. This will help them complete the migration, moving their existing code to Drupal. These are the tools the content editors need to make their website beautiful and effective. If your content editors are more familiar with writing javascript and css here’s how to enable them to keep doing that.
To make this happen, first make a raw field formatter.
- Go to Configuration > Content authoring > Text formats and editors.
- Add a new text format called “Raw”. None of the filters should be enabled since this will be raw output.
Adding in raw text format
AND…No filters enabled!
Since our client wants to add raw css and javascript to landing pages, we will create a field on the ‘landing page’ content type. It will be Text (formatted, long) and label “Inline CSS”. We will limit it to just one on the page.
Add field inline css
Have it use the Raw text format from the last step. You can limit the field to only this format by installing the package
Composer require drupal/allowed_formats
Be sure to check the “Raw” box on the field page and save it.
Now make sure our field is being output.
- Go to Admin > Structure > Types > Manage > Landing page > Display > Full
- Make sure it is enabled and the label is hidden. It should be output in the default format.
Making sure inline css is displayed
Visit a landing page content form by going to Manage > Content > Add content > Landing Page, and put some real css in our new field:
Adding map background raw
We also provide a WYSIWYG place to enter HTML. In this case we need some HTML, perhaps a div, with class=‘map’.
We’re not finished yet! We need to provide a twig template. Look at the output HTML. We get:
<span class="c"><!-- THEME DEBUG --></span><span class="c"><!-- THEME HOOK: 'field' --></span><span class="c"><!-- FILE NAME SUGGESTIONS:* field--node--field-inline-css--landing-page.html.twig* field--node--field-inline-css.html.twig* field--node--landing-page.html.twig* field--field-inline-css.html.twigx field--text-long.html.twig* field.html.twig--></span><span class="c"><!-- BEGIN OUTPUT from 'core/themes/classy/templates/field/field--text-long.html.twig' --></span><span class="nt"><div</span> <span class="na">data-quickedit-field-id=</span><span class="s">"node/589/field_inline_css/en/full"</span> <span class="na">class=</span><span class="s">"clearfix text-formatted field field--name-field-inline-css field--type-text-long field--label-hidden field__item"</span><span class="nt">></span>.map {background: url(http://www.example.com/assets/images/background-images/banner-landing-page/map.png) center no-repeat;padding-top: 80px;min-height: 350px;}<span class="nt"></div></span><span class="c"><!-- END OUTPUT from 'core/themes/classy/templates/field/field--text-long.html.twig' --></span>
in our output! Notice the <div>
surrounding our CSS! We don’t want that! So it’s time to create a Twig template without extra div’s. One that will output raw CSS.
We will go from this (notice all the extra <div>
s)
{% if label_hidden %} {% if multiple %} <span class="nt"><div</span><span class="err">{{</span> <span class="na">attributes</span><span class="err">.</span><span class="na">addClass</span><span class="err">(</span><span class="na">classes</span><span class="err">,</span> <span class="err">'</span><span class="na">field__items</span><span class="err">')</span> <span class="err">}}</span><span class="nt">></span> {% for item in items %} <span class="nt"><div</span><span class="err">{{</span> <span class="na">item</span><span class="err">.</span><span class="na">attributes</span><span class="err">.</span><span class="na">addClass</span><span class="err">('</span><span class="na">field__item</span><span class="err">')</span> <span class="err">}}</span><span class="nt">></span>{{ item.content }}<span class="nt"></div></span> {% endfor %} <span class="nt"></div></span> {% else %} {% for item in items %} <span class="nt"><div</span><span class="err">{{</span> <span class="na">attributes</span><span class="err">.</span><span class="na">addClass</span><span class="err">(</span><span class="na">classes</span><span class="err">,</span> <span class="err">'</span><span class="na">field__item</span><span class="err">')</span> <span class="err">}}</span><span class="nt">></span>{{ item.content }}<span class="nt"></div></span> {% endfor %} {% endif %}{% else %} <span class="nt"><div</span><span class="err">{{</span> <span class="na">attributes</span><span class="err">.</span><span class="na">addClass</span><span class="err">(</span><span class="na">classes</span><span class="err">)</span> <span class="err">}}</span><span class="nt">></span> <span class="nt"><div</span><span class="err">{{</span> <span class="na">title_attributes</span><span class="err">.</span><span class="na">addClass</span><span class="err">(</span><span class="na">title_classes</span><span class="err">)</span> <span class="err">}}</span><span class="nt">></span>{{ label }}<span class="nt"></div></span> {% if multiple %} <span class="nt"><div</span> <span class="na">class=</span><span class="s">"field__items"</span><span class="nt">></span> {% endif %} {% for item in items %} <span class="nt"><div</span><span class="err">{{</span> <span class="na">item</span><span class="err">.</span><span class="na">attributes</span><span class="err">.</span><span class="na">addClass</span><span class="err">('</span><span class="na">field__item</span><span class="err">')</span> <span class="err">}}</span><span class="nt">></span>{{ item.content }}<span class="nt"></div></span> {% endfor %} {% if multiple %} <span class="nt"></div></span> {% endif %} <span class="nt"></div></span>{% endif %}
And we should do three things:
- Remove all
<div>
tags, - Send it through a raw filter, and
- Surround it with
<style>
tags so we will go to this >
<span class="nt"><style></span><span class="p">{</span><span class="err">%</span> <span class="err">if</span> <span class="err">label_hidden</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">if</span> <span class="err">multiple</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">for</span> <span class="err">item</span> <span class="err">in</span> <span class="err">items</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">{</span> <span class="err">item.content|raw</span> <span class="p">}</span><span class="err">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">endfor</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">else</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">for</span> <span class="err">item</span> <span class="err">in</span> <span class="err">items</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">{</span> <span class="err">item.content|raw</span> <span class="p">}</span><span class="err">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">endfor</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">endif</span> <span class="err">%</span><span class="p">}</span><span class="p">{</span><span class="err">%</span> <span class="err">else</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">if</span> <span class="err">multiple</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">endif</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">for</span> <span class="err">item</span> <span class="err">in</span> <span class="err">items</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">{</span> <span class="err">item.content|raw</span> <span class="p">}</span><span class="err">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">endfor</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">if</span> <span class="err">multiple</span> <span class="err">%</span><span class="p">}</span> <span class="p">{</span><span class="err">%</span> <span class="err">endif</span> <span class="err">%</span><span class="p">}</span><span class="p">{</span><span class="err">%</span> <span class="err">endif</span> <span class="err">%</span><span class="p">}</span><span class="nt"></style></span>
Then we get in output:
<span class="c"><!-- THEME DEBUG --></span><span class="c"><!-- THEME HOOK: 'field' --></span><span class="c"><!-- FILE NAME SUGGESTIONS:x field--node--field-inline-css--landing-page.html.twig* field--node--field-inline-css.html.twig* field--node--landing-page.html.twig* field--field-inline-css.html.twig* field--text-long.html.twig* field.html.twig--></span><span class="c"><!-- BEGIN OUTPUT from 'themes/custom/example/templates/field/field--node--field-inline-css--landing-page.html.twig' --></span><span class="nt"><style></span><span class="nc">.map</span> <span class="p">{</span><span class="nl">background</span><span class="p">:</span> <span class="sx">url(http://www.example.com/assets/images/background-images/banner-section-landing-page/map.png)</span> <span class="nb">center</span> <span class="nb">no-repeat</span><span class="p">;</span><span class="nl">padding-top</span><span class="p">:</span> <span class="m">80px</span><span class="p">;</span><span class="nl">min-height</span><span class="p">:</span> <span class="m">350px</span><span class="p">;</span><span class="p">}</span><span class="nt"></style></span><span class="c"><!-- END OUTPUT from 'themes/custom/example/templates/field/field--node--field-inline-css--landing-page.html.twig' --></span>
Tada! The CSS shows up ready to use on the page! The same technique can be used to allow content editors to put JavaScript on the page! Instead of putting <style>
tags around the template, make it <script>
tags instead.
Make sure you meet your content editors where they are, give them tools they can use but don’t use this technique with novice or non-technical content editors.