code icon indicating copy to clipboard operation
code copied to clipboard

How can I set language for code?

Open nguyentranchung opened this issue 6 years ago • 17 comments

Ex: php, css, js, ...

nguyentranchung avatar Apr 08 '19 23:04 nguyentranchung

this would be awesome feature to have the ability to set language per snippet and be able to highlight the code with third party tools like highlight.js

3alampro avatar Apr 09 '19 22:04 3alampro

+1 for this feature!

ddimitrioglo avatar Apr 29 '19 19:04 ddimitrioglo

Under the hood simple textarea, adding third party syntax highlight like highlight.js not resolve the issue. Need add code editor like codemirror for highlight on input. I think for this better create new block tool or include code editor into current block tool.

dimensi avatar Apr 30 '19 09:04 dimensi

If anyone needs the solution , i have the custom version ready for production , with support for language as seen in below screenshots 😄

2019-05-11_13-53-54 2019-05-11_13-54-25

Update#1 : Works perfectly with language- tag for Prism.js as well , the sent data for language is in format of alias used for prism.js

dukesx avatar May 11 '19 08:05 dukesx

I have a solution without highlighting the code. I'm a new learner. Here is the code:

/**
 * Build styles
 */
require('./codeplus.css').toString();

class CodePlus {

    static get toolbox() {
        return {
            icon: '<svg width="14" height="14" viewBox="0 -1 14 14" xmlns="http://www.w3.org/2000/svg" > <path d="M3.177 6.852c.205.253.347.572.427.954.078.372.117.844.117 1.417 0 .418.01.725.03.92.02.18.057.314.107.396.046.075.093.117.14.134.075.027.218.056.42.083a.855.855 0 0 1 .56.297c.145.167.215.38.215.636 0 .612-.432.934-1.216.934-.457 0-.87-.087-1.233-.262a1.995 1.995 0 0 1-.853-.751 2.09 2.09 0 0 1-.305-1.097c-.014-.648-.029-1.168-.043-1.56-.013-.383-.034-.631-.06-.733-.064-.263-.158-.455-.276-.578a2.163 2.163 0 0 0-.505-.376c-.238-.134-.41-.256-.519-.371C.058 6.76 0 6.567 0 6.315c0-.37.166-.657.493-.846.329-.186.56-.342.693-.466a.942.942 0 0 0 .26-.447c.056-.2.088-.42.097-.658.01-.25.024-.85.043-1.802.015-.629.239-1.14.672-1.522C2.691.19 3.268 0 3.977 0c.783 0 1.216.317 1.216.921 0 .264-.069.48-.211.643a.858.858 0 0 1-.563.29c-.249.03-.417.076-.498.126-.062.04-.112.134-.139.291-.031.187-.052.562-.061 1.119a8.828 8.828 0 0 1-.112 1.378 2.24 2.24 0 0 1-.404.963c-.159.212-.373.406-.64.583.25.163.454.342.612.538zm7.34 0c.157-.196.362-.375.612-.538a2.544 2.544 0 0 1-.641-.583 2.24 2.24 0 0 1-.404-.963 8.828 8.828 0 0 1-.112-1.378c-.009-.557-.03-.932-.061-1.119-.027-.157-.077-.251-.14-.29-.08-.051-.248-.096-.496-.127a.858.858 0 0 1-.564-.29C8.57 1.401 8.5 1.185 8.5.921 8.5.317 8.933 0 9.716 0c.71 0 1.286.19 1.72.574.432.382.656.893.671 1.522.02.952.033 1.553.043 1.802.009.238.041.458.097.658a.942.942 0 0 0 .26.447c.133.124.364.28.693.466a.926.926 0 0 1 .493.846c0 .252-.058.446-.183.58-.109.115-.281.237-.52.371-.21.118-.377.244-.504.376-.118.123-.212.315-.277.578-.025.102-.045.35-.06.733-.013.392-.027.912-.042 1.56a2.09 2.09 0 0 1-.305 1.097c-.2.323-.486.574-.853.75a2.811 2.811 0 0 1-1.233.263c-.784 0-1.216-.322-1.216-.934 0-.256.07-.47.214-.636a.855.855 0 0 1 .562-.297c.201-.027.344-.056.418-.083.048-.017.096-.06.14-.134a.996.996 0 0 0 .107-.396c.02-.195.031-.502.031-.92 0-.573.039-1.045.117-1.417.08-.382.222-.701.427-.954z" /> </svg>',
            title: 'Code'
        }
    }

    static get contentless() {
        return true;
    }

    static get enableLineBreaks(){
        return true;
    }

    static get DEFAULT_CODE_PLACEHOLDER() {
        return 'Enter some code';
    }

    static get DEFAULT_LANGUAGE_PLACEHOLDER() {
        return 'Select a format';
    }

    static get DEFAULT_FORMAT_CONFIG() {
        return ['Python', 'Matlab', 'R', 'Javascript', 'C', 'HTML'];
    }

    get CSS() {
        return {
            baseClass: this.api.styles.block,
            wrapper: 'cdx-codeplus',
            input: this.api.styles.input,
            language: 'cdx-codeplus__language',
            textarea: 'cdx-codeplus__textarea'
        };
    }

    constructor({
        data,
        config,
        api
    }) {
        this.api = api;

        this.textPlaceholder = config.textPlaceholder || CodePlus.DEFAULT_CODE_PLACEHOLDER;
        this.languagePlaceholder = config.languagePlaceholder || CodePlus.DEFAULT_LANGUAGE_PLACEHOLDER;
        this.format = config.format || CodePlus.DEFAULT_FORMAT_CONFIG;

        this.data = {
            language: data.language || '',
            text: data.text || ''
        };
    }

    render() {
        const container = this._make('div', [this.CSS.baseClass, this.CSS.wrapper]);
        const text = this._make('textarea', [this.CSS.input, this.CSS.textarea], {
            contentEditable: true,
            innerHTML: this.data.text
        });
        const selectwrapper=this._make('div',this.CSS.input);
        const language = this._make('select', this.CSS.language);


        text.dataset.placeholder = this.textPlaceholder;
        language.dataset.placeholder = this.languagePlaceholder;

        const format = this.format;


        for (let f in format) {
            let option = document.createElement('option');
            let v = document.createAttribute('value');
            let t = document.createTextNode(format[f]);
            v.value = format[f];
            option.appendChild(t);
            option.setAttributeNode(v);
            language.appendChild(option);
        }

        language.value = this.data.language;
        selectwrapper.appendChild(language);
        container.appendChild(selectwrapper);
        container.appendChild(text);

        return container;
    }

    onPaste(event) {
        const content = event.detail.data;
        this.data = {
            text: content.textContent
        };
    }

    save(codeElement) {
        const text = codeElement.querySelector(`.${this.CSS.textarea}`);
        const language = codeElement.querySelector(`.${this.CSS.language}`);
        let index = language.selectedIndex;

        return Object.assign(this.data, {
            text: text.value,
            language: language.options[index].value
        });
    }

    static get sanitize() {
        return {
            language: {},
            text: {}
        };
    }

    _make(tagName, classNames = null, attributes = {}) {
        let el = document.createElement(tagName);

        if (Array.isArray(classNames)) {
            el.classList.add(...classNames);
        } else if (classNames) {
            el.classList.add(classNames);
        }

        for (let attrName in attributes) {
            el[attrName] = attributes[attrName];
        }

        return el;
    }

    static get pasteConfig() {
        return {
            tags: ['pre'],
        };
    }
}
module.exports = CodePlus;

and the style file:

.cdx-codeplus__textarea {
    min-height: 200px;
    font-family: Menlo, Monaco, Consolas, Courier New, monospace;
    color: #41314e;
    line-height: 1.6em;
    font-size: 12px;
    background: #f8f7fa;
    border: 1px solid #f1f1f4;
    box-shadow: none;
    white-space: pre;
    word-wrap: normal;
    overflow-x: auto;
    resize: vertical;
}

.cdx-codeplus__language {
    width: 100%;
    cursor: pointer;
    padding-right: 2em;
    border: none;
    background: transparent;
    background-image: none;
    -webkit-appearance: none;
    -moz-appearance: none;
    appearance: none;
    text-indent: 0.01px;
    text-overflow: '';
}

mmhubcn avatar May 13 '19 09:05 mmhubcn

@dukesx, and where is your solution? Give us the link

zhkuskov avatar May 20 '19 09:05 zhkuskov

@dukesx, dude where is the solution for this implementation? Can you please share the link here, would really appreciate it.

utkarshrai003 avatar Oct 18 '19 20:10 utkarshrai003

Sorry guys, I have been really busy with other projects. This implementation was actually for my cms project that I made and which went down in flames. Now I am remaking that cms, and I will be using editor.js with same implementation. I will upload the snippet here sometime later when I am out of design phase or you can follow my project "Write Grand CMS" to see updates on what's going on.

dukesx avatar Oct 18 '19 20:10 dukesx

@PolinaShneider any update on this one :-D

palashgupta98 avatar Feb 12 '20 06:02 palashgupta98

Hi all,

Check out new CodeBox tool. Has support for multiple languages and multiple themes.

If you use React in your project, the ERR tool might be useful to you. It helps with rendering your saved block data into styled and flexible react components.

PS: I build both tools for use on a project i'm currently working on, so thought others might benefit too. So, any questions feel free to ask directly.

Enjoy!

dev-juju avatar Feb 13 '20 16:02 dev-juju

I've created a codemirror editor.js, with a selection of the code:

image

https://github.com/alexiej/editorjs-codemirror

There are couple of bugs from editor.js:

  • Because of the blinking cursor, the content change every time class and style for the div and the editor.js raise the event updated, and OnChange() every second when it is focused.

I've done with this on the editor.js to modificationsObserver.ts. Now OnChange only raises when there is a real change. https://github.com/alexiej/editor.js/commit/9902f69c7b5aac8ff26ee0e15ad38f13b412001c

alexiej avatar Feb 20 '20 13:02 alexiej

I have created a fork from this repo, that lets you select the language code (supported by prismJs). https://github.com/paraswaykole/editor-js-code

paraswaykole avatar May 29 '20 20:05 paraswaykole

If anyone needs the solution , i have the custom version ready for production , with support for language as seen in below screenshots 😄

2019-05-11_13-53-54 2019-05-11_13-54-25

Update#1 : Works perfectly with language- tag for Prism.js as well , the sent data for language is in format of alias used for prism.js

Can you share the code

siddhuphp avatar Jun 28 '20 10:06 siddhuphp

I have added a pull request here for this issue which allows the user to add their own language with a custom label

epndavis avatar Aug 18 '20 16:08 epndavis

I found great solution here https://github.com/calumk/editorjs-codeflask image Codeflask used as an editor

Hydrock avatar Dec 07 '22 17:12 Hydrock

Another way to do it is by extending the CodeTool class without touching its core implementation:

import EditorJS from '@editorjs/editorjs'; 
import CodeTool from '@editorjs/code';

//========================================
// Class Extension
//========================================
class Code extends CodeTool {
  /**
  * Render plugin`s main Element and fill it with saved data
  */
  constructor({ data, config, api, readonly }) {
    super({ data, config, api, readonly });
    
    if (config.languages !== undefined && config.languages?.length > 0) {
      this.languages = config.languages;
      
      this.CSS.dropdown = "ce-code__dropdown";
      console.log(config.languages)
      
      this.nodes.dropdown = null;
      
      this.data.language = data.language;
      
      this.nodes.holder = this.drawView();
    }
  }
  
  /**
  * Create Tool's view
  */
  drawView() {
    // Create the default tool view
    const wrapper = super.drawView();
    const textarea = wrapper.querySelector("textarea");
    
    // Then, add dropdown menu
    if (this.languages !== undefined && this.languages.length > 0) {
      const dropdown = document.createElement("select");
      dropdown.classList.add(this.CSS.dropdown, this.CSS.input);
      dropdown.value = this.data.language;
      
      this.languages.forEach((lang) => {
        let option = document.createElement("option");
        option.classList.add(`${this.CSS.dropdown}__option`);
        option.textContent = lang;
        dropdown.appendChild(option);
      });
      
      wrapper.insertBefore(dropdown, textarea);
      this.nodes.dropdown = dropdown;
    }
    return wrapper;
  }
  
  /**
  * Extract Tool's data from the view
  */
  save(codeWrapper) {
    const data = super.save(codeWrapper);
    if (this.languages !== undefined && this.languages.length > 0)
      return {
        ...data,
        language: codeWrapper.querySelector("select")?.value,
      };
    else return data;
  }
}

//========================================
// Usage
//========================================
const editor = new EditorJS({
  holder: 'editorjs', 
  tools: {
    code: {
      class: Code,
      config: {
        languages: ["js", "csharp"]
      }
    }
  }
});

Here is the css styles for the dropdown menu:

.ce-code__dropdown {
  -webkit-appearance: none;
  -moz-appearance: none;
  -ms-appearance: none;
  -o-appearnance: none;
  appearnance: none;
  margin-bottom: 8px;
}

Result

image

Output

image

Al3bad avatar Feb 08 '24 13:02 Al3bad