npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

gulp-syntaxhighlighter

v1.0.6

Published

Apply syntax highlighter at build time - if hiding/showing is required adds small amount of javascript and css

Downloads

13

Readme

gulp-syntaxhighlighter

Is a gulp plugin for running syntaxhighlighter v3.0.90 at build time rather than in the browser at run time.

Syntaxhighlighter was downloaded from here but had to be rebuilt see here).

The documentation for syntax highlighter can be found here and also here for config and global parameters.

By generating at runtime there is a guarantee that your code blocks will display as desired when javascript is not available such as when disabled in the browser or when contents are being read by an rss reader.

Essentially the plugin functions the same as when run in the browser. The only difference is that since we are not relying upon the presence of javascript, syntax highlighter will not be present in the stream output and any dynamic functionality it provided will no longer function. There are two pieces of such functionality in syntax highlighter, the ability to load the syntax highlighted elements as hidden and then to show them. The other is a text area for copying the code.

I have provided more advanced toggle functionality which if opted in for will result in some additional css and javascript added to File.contents. As noted this is not necessary for the html to be syntax highlighted. See how to specify later.

The plugin and options

The options shown below are a combination of the config, parameters and theming of syntax highlighter and options that are specific to the toggle functionality. It is necessary to provide the toggleConfig property object for toggle functionality.

//global - can specify in html for each element to be transformed ( apart from placement )
export interface ToggleConfigMessage{
    //---------- can be set in html
    //default Never
    when?:"Always"|"Hidden"|"Never",
    //default ""
    prefix?:string,
    //default ""
    prefixShow?:string,
    //default ""
    prefixHide?:string,
    //default false
    useTitle?:boolean,
    //default ""
    hideMessage?:string,
    //default ""
    showMessage?:string,
    //default ""
    message?:string,

    //---------- cannot be set in html
    //default Right
    placement?:"Right"|"Below"
}
//each default is the same as the property name
export interface ClassNames{
    toggleContainer?:string,
    toggleText?:string,
    toggle?:string,
    showToggle?:string,
    hideToggle?:string,
    isShowing?:string
}
export interface ToggleConfig{
    //global toggleState to apply to all
    toggleState?:"Show"|"Hide",
    //global to apply to all
    message?:ToggleConfigMessage,
    //default creates two svg elements, one for hide one for show
    //SIGNATURE IS - function createToggleElement(show)
    createToggleFn?:string,
    classNames?:ClassNames,
    //for replacing the css that would normally be applied for the toggle
    customCss?:string
}
export interface SyntaxHighlighterOptions{
    // default - see later
    isPartialFn?:(html:string,file:File)=>boolean

    //default true - for the js being used
    useMinifiedSyntaxHighlighter?:boolean
    //default true - for theme css,additionalCss, toggle js/css
    minifiedOutput?:boolean

    //default Default
    theme?:string,
    //actual css instead of theme by name above
    customTheme?:string,
    //in addition to theme|customTheme
    additionalCss?:string,

    //syntax highlighter parameters
    globalParams?:{
        "class-name"?:string,
        "first-line"?:number
        'pad-line-numbers' ?: boolean,
        'highlight' ?: number[],
        'title' ?: string,
        'smart-tabs' : boolean,
        'tab-size' : number,
        'gutter' : boolean,
        'auto-links' : boolean,
        'unindent' : boolean,
        'html-script' : false
    }
    //syntax highlighter config ( to merge with the defaults )
    config?: {
        space ?: string,
        useScriptTags? : boolean,
        bloggerMode? : boolean,
        stripBrs? : boolean,
        tagName? : string,
    }
    //if not present then no toggle js/css
    toggleConfig?:ToggleConfig
}
export function syntaxHighlighter(options?:SyntaxHighlighterOptions):Transform

Options explanation

useMinifiedSyntaxHighlighter - this you would leave as the default unless you wanted to debug the plugin

isPartialFn - This affects the generated output. For some circumstances File.contents could be loose html ( such as for blogger ) and it is necessary to retain that in the output. If you are not happy with my attempt at it, replace with your own function.

private isPartial=function(html:string,file:File){
        const firstTag=html.substring(html.indexOf("<"),html.indexOf(">")).toLowerCase();
        return !(firstTag.startsWith("<!doctype")||firstTag.startsWith("<html")||firstTag.startsWith("<head")||firstTag.startsWith("<body"));
    }

minifiedOutput - This defaults to true and if true any css or js added in the output will be minified.

theme - this is the name of a theme. Possible values are Default, Django, Eclipse, Emacs, FadeToGrey, MDUltra, Midnight and RDark.

customTheme - this is css for your own custom theme instead of using the theme name above

additionalCss - this is css in addition to either the built in themes or your own custom theme.

Toggle functionality

Toggling is achieved by a small script added to the body that includes the ToggleConfig.createToggleFn string value. The signature is as follows and the default creates two svgs :

function createToggleElement(show:boolean):Node

There are two ways to indicate that an element requires toggling either globally through the ToggleConfig.toggleState option or by specifying through a data-toggleState attribute on the element. Note that the latter method supercedes the first. If there is no toggle state determined for an element through this process then there will be no toggling and the syntax highlighter transformed element will display and there will be no icon generated to hide/show it.

If a toggle state has been specified then toggle elements are created using the createToggleFn with click handlers to hide/show the syntax highlighted element. Initial state as determined by the toggle state.

Regardless of the createToggleFn used the structure is always the same and the class names are applied using toggleConfig.classNames. Both have the toggle class and either of the showToggle or hideToggle classes.

<div class="toggleContainer">
    <ToggleElementShow|Hide/>
    <!-- Possibly a text element -->
    <div|span class="toggleText">Message text depending upon settings</div|span>
</div>

The toggle container will also get the isShowing class when the element is being shown.

The text element will be present depending upon the 'when' setting. If 'Always' it will always be present, if 'Hidden' only when hidden. It will be a span if the placement is "Right", if "Below" it will be a div.

The specification of placement is through the ToggleConfigMessage.placement option. The when setting is like the toggleState in that it can be set globally or on each element through a data- attribute. All of the ToggleConfigMessage properties other than placement can be set in this manner. The format is data-togglepropertyname. Data-toggle attributes supercede the global values.

The code below determines the message that will be shown. There is a show message and a hide message which can then have a prefix applied. The unprefixed message can come from the syntax highlighter title 'class attribute' with useTitle or showMessage|hideMessage as appropriate defaulting to the message config value when no showMessage|hideMessage. The prefixing works in the same manner.


function getMessages(message:ToggleConfigMessage,syntaxhighlighter:HTMLElement):Messages{
        let showMessage="";
        let hideMessage="";

        if(message.useTitle){
            const caption=getCaption(syntaxHighlighter)

            showMessage=caption;
            hideMessage=caption;
        }else{
            showMessage=message.showMessage!==""?message.showMessage:message.message;
            hideMessage=message.hideMessage!==""?message.hideMessage:message.message;
        }

        showMessage=(message.prefixShow!==""?message.prefixShow:message.prefix) + showMessage;
        hideMessage=(message.prefixHide!==""?message.prefixHide:message.prefix) + hideMessage;

        return {
            hiddenMessage:hideMessage,
            showMessage:showMessage
        }
    }

There is a small amount of css applied ( using the ToggleConfig.classNames ).

private getDefaultToggleCss(classNames:ClassNames):string{
        const svgSize="1em";
        return `.${classNames.toggle}{ stroke:#000;top:0.125em;position:relative;height:${svgSize};width:${svgSize} }
        .${classNames.toggleText}{ font-size:${svgSize}}
        `
    }

If this is not desired ( perhaps you want to change the stroke colour or perhaps you provided your own createToggleFn ) then you can supply your own with the ToggleConfig.customCss.

Testing / Demo

This has not yet been tested in the traditional sense but you can download the project from github and run gulp demo. Demo/dest will have two html files that have been transformed by the plugin each with different plugin options chosen to cover all functionality.

To Do

Make placement overridable in the html ? I had removed config strings as at first glance did not think that they were applicable. Check in depth. Because the syntax highlighter project is old ( it has been replaced with version 4 which failed to build miserably ) there are two css settings that have vendor prfixes that can be replaced. This needs to be done here. Provide code copying functionality. Facility to provide custom brushes.

Do tests.