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

@knighttower/adaptive

v3.10.0

Published

Stand-alone or with Vue Hybrid mode Js helper library to help with adaptive layouts

Downloads

91

Readme

(This docs still under construction, see "example" and "test" folders for extensive examples of usage)

Adaptive.js: Enhancing Layout Adaptability Beyond CSS

It Uses/Offers:

  • Teleport - Teleport element base on the css breakpoint and reverse it back to its original place at command. It can be used as a drop-in into vanilla js projects or along with Vue or React. See stand-alone or docs for more info here-> https://github.com/knighttower/JsTeleport
  • Dynamic Class and Attribute Management - Add remove classes or inline styles per breakpoint
  • Execute JS function at CSS breakdowns - Helps to have specific code executing only when needed
  • Vue and React support
  • Supports all modern browsers
  • Built as a module, can be used as a drop-in (for browser only use) or as a plugin for Vue or React.

NPM published pages-build-deployment release version

Contents

Installation:

npm stand-alone

npm i @knighttower/adaptive

Or as part of the monorepo

npm i knighttower

Drop-in from CDN (no build step)

<script src="https://cdn.jsdelivr.net/npm/@knighttower/adaptive@latest/dist/browser/Adaptive.min.js"></script>
// or if need to use ESM, CJS, UMD, via browser //ESM
<script type="module">
    import { Adaptive } from 'https://esm.run/@knighttower/adaptive@latest/index.js';
</script>
// UMD
<script src="https://cdn.jsdelivr.net/npm/@knighttower/adaptive@latest/dist/umd/Adaptive.min.js"></script>

Files

| File | Size | | --------------- | ------ | | /Adaptive.js | 18 KiB | | /Adaptive.js.br | 9 KiB | | /Adaptive.js.gz | 9 KiB |

By default the "import" or "require", will load the indexes automatically. But, in case of wanting to use individual files or other specific formats, all Files are available in the dist folder as ESM, CJS, AMD, IIFE, Browser, UMD and System formats. For ESM + JS 'next', use the files in the src folder or import directly from the index.js (index.cjs.js for commonJS) file.

Usage

🙋‍♂️ see the "example" folder for all 'usage' implementation examples, including Browser, Vue and React. --> here

As a module for Vue

const { createApp } = Vue;
import Adaptive from 'adaptive.js';
const App = createApp({});
// NOTE: that adaptive will self register, do not use App.use()
// Adaptive does that automatically
Adaptive.useVue(App);
App.mount('#app');

As stand-alone script in you project

<script src="//cdn.jsdelivr.net/npm/@knighttower/[email protected]/dist/Adaptive.min.js"></script>
<script>
    $adaptive.init();
</script>

As a module for React

import React from 'react';
import ReactDom from 'react-dom';
import Adaptive from 'adaptive.js';
// NOTE: that adaptive will self register, do not use App.use()
// Adaptive does that automatically
Adaptive.useReact(React);
Adaptive.init();
// ... other code

Implementation examples

  • Simple String - It just teleports, does not specify breakpoints and location

    data-teleport="#hello"
    data-teleport-before="#hello"
    data-teleport-after="#hello"

<div data-teleport="#hello">
    <span style="background:#083d39; color: white;"
        >2 Static element that will teleport "hello" with data attribute using an array like string</span
    >
</div>

  • Dot notation - Allows to specify multiple breakpoints and locations
    data-adaptive=[command].[breakpoint].[to|from|after](target_id_or_class)
    the "|" is "OR": [breakpoint]|[breakpoint]
<!-- Teleporting only at mobile and desktop, all others will be set back to original position -->
<div data-adaptive="teleport.desktop|mobile.before(#hello)">
    <span style="background:#1f2252; color: white;"
        >Static element that will teleport at "desktop or mobile" right above "static hello"</span
    >
</div>

  • Object like
    data-adaptive={'command':{breakpoint:{[to|before|after]:target_id_or_class}}}
<div
    data-adaptive="{'teleport':{
	'tablet': {
	'to':'.lazy'
}
}}"
>
    <span>Static element that will teleport at "tablet" size to a Vue driven element when visible</span>
</div>

  • Array like
    data-teleport=[[after|before], target_id_or_class]
<div data-teleport="[after,'#hello']">
    <span style="background:#083d39; color: white;"
        >2 Static element that will teleport "hello" with data attribute using an array like string</span
    >
</div>

  • Inside Vue
    Use it to trigger functions or methods at specific breakpoints.

  • Array, object or dot notation works inside Vue, just use "v-adaptive=..." or " v-teleport-to..."

<div v-teleport-to="'#hello'">Getting teleported (teleport) from the component to "static Hello"</div>

// Or in code
<script>
    //some code ...
 mounted() {
	this.Adaptive.if('tablet', function() {
            // code
	});
	// or
	this.Adaptive.if('tablet', this.nameOfMethod);
	// ...See more use cases in the example files!

    //or

    this.Adaptive.if('tablet', function () {
        // code... run only once
    }).onlyOnce();
}
</script>

Features

  • Markup Integration: Easily integrate directly into your HTML markup or within JavaScript.
  • Breakdown-Specific Class and Style Management: Modify classes and styles dynamically based on screen size.
  • Custom Queries: Define custom media queries to tailor layouts to specific needs.
  • Hybrid Mode: Combine Adaptive.js with Vue or React, allowing static and dynamic content to coexist seamlessly.

API

getMinMaxQueries()

  • Description: Returns all available min-max queries.
  • Example:
    const minMaxQueries = Adaptive.getMinMaxQueries();

getExpQueries()

  • Description: Returns all available expression queries.
  • Example:
    const expQueries = Adaptive.getExpQueries();

registerElement(elementOrSelector, data)

  • Description: Registers an element for adaptive behavior.
  • Example:
    Adaptive.registerElement('#myElement', { directive: 'value' });

addQueryMinMax(id, min, max)

  • Description: Registers a custom min-max query.
  • Example:
    Adaptive.addQueryMinMax('customQuery', 500, 800);

addQueryExpression(id, query)

  • Description: Registers a custom expression query.
  • Example:
    Adaptive.addQueryExpression('customQuery', 'screen and (max-width: 600px)');

if(breakdownId, callback)

  • Description: Executes a callback function based on the media query match.
  • Example:
    Adaptive.if('mobile', () => { console.log('Mobile view'); });

init()

  • Description: Initializes Adaptive.js after the DOM has loaded.
  • Example:
    Adaptive.init();

useVue(Vue, hybrid)

  • Description: Configures Adaptive.js for use with Vue.
  • Example:
    Adaptive.useVue(VueInstance, true);

useReact(React, hybrid)

  • Description: Configures Adaptive.js for use with React.
  • Example:
    Adaptive.useReact(ReactInstance, true);


Note: For a complete overview, refer to the examples in the example and test directories and explore the live demos.

Directly in markup

<!-- Example of only adding styles per breakdown -->
<div class="inline-styling" data-adaptive="{'addStyle':{'tablet':'color: blue;','desktop':'color: green;'}}">
    Inline styling per device
</div>

<!-- Example of using custom queries -->
<div
    id="customQueries"
    data-adaptive="addClass[(min-width: 900px) and (max-width: 1599px)](custom-class-at-custom-breakdown)"
>
    Custom Expressions via string like Directive
</div>

<!-- Teleporting -->
<div data-teleport="{'after':'#customQueries'}">
    <span style="background:#067e74; color: white;"
        >Static element that will teleport "customQueries" with data attribute in hybrid mode</span
    >
</div>

Along with JS code

<script>
    // Target or register specific elements
    $adaptive.registerElement('#hello', {
        addClass: {
            mobile: 'a-class-added-only-for-mobiles',
        },
    });

    // OR
    // Run functions at specific layouts or special js needs
    $adaptive.if('tablet', function () {
        // code to execute here
    });

    // ...See more use cases in the example files!
</script>

Adaptive.js works as stand-alone out-of-the-box, but it also works along side integrated with Vue (hybrid mode) as it offers direct integration with it (as a plugin and a directive) or with React (see the examples folder in the source code). This approach will help to cover applications that have both dynamic and static markup ("static" = not accessible either by Vue/React or its components) but it still needs to play nice with the responsive/adaptive style for any screen size.

For frameworks like React, Adaptive would work with them in a slightly different manner and not as tightly coupled as it could be with Vue.

- Example as Stand Alone:

<html>
    <head> </head>
    <body>
        <section id="app">
            <!-- Example of only adding styles per breakdown -->
            <div
                class="inline-styling"
                data-adaptive="{'addStyle':{'tablet':'color: blue;','desktop':'color: green;'}}"
            >
                Inline styling per device
            </div>

            <br />
            <!-- Teleporting -->
            <div data-teleport="{'after':'#customQueries'}">
                <span style="background:#067e74; color: white;"
                    >Static element that will teleport "customQueries" with data attribute in hybrid mode</span
                >
            </div>
        </section>

        <!-- JS -->
        <script src="../../dist/Adaptive.js"></script>
        <script>
            $adaptive.init();
        </script>
    </body>
</html>

Example as hybrid mode and advance modes:

// In App.js
const { createApp } = Vue;
import Adaptive from '../src/Adaptive.js';

const App = createApp({});

// Optional: Add custom media query (min px, max px) settings (min max)
Adaptive.addQueryMinMax('myCustomMediaQueryName', 900, 1400);
// Optional: Add a custom media query expression (it accepts any valid media query)
Adaptive.addQueryExpression('myCustomMediaQueryName2', '(min-width: 900px)');
// Instanciate right after the app and before the components
Adaptive.useVue(App);
// Do components and other stuff right after
App.component('hello', hello);
// Finally mount the app
App.mount('#app');

// -----------------------------------
// In Component XX
// <!-- Example of directive binding with Adaptive directly in the element  -->
<div v-adaptive="{ addClass: { desktop: 'adds-this-class-xxxx' } }">
    Using Directive inside Vue component
</div>
<div v-teleport-to="'#hello'">Getting teleported (teleport) from the component to "static Hello"</div>

// Or in code
<script>
.... mounted() {
	this.Adaptive.if('tablet', function() {
            // code
	});
	// or
	this.Adaptive.if('tablet', this.nameOfMethod);
	// ...See more use cases in the example files!
}
</script>

API

getMinMaxQueries()

Returns: Object
Description: Returns all available min-max queries.

Example

const minMaxQueries = Adaptive.getMinMaxQueries();

getExpQueries()

Returns: Object
Description: Returns all available expression queries.

Example

const expQueries = Adaptive.getExpQueries();

registerElement(elementOrSelector: String | Object, data: Object)

Parameters:

  • elementOrSelector (String | Object): Element or selector string.
  • data (Object): Optional, used mostly for Vue.

Returns: Void
Description: Registers an element for adaptive behavior.

Example

Adaptive.registerElement('#myElement', { directive: 'someValue' });

addQueryMinMax(id: String, min: Number, max: Number)

Parameters:

  • id (String): Identifier for the query.
  • min (Number): Minimum value.
  • max (Number): Maximum value.

Returns: Void
Description: Registers a custom min-max query.

Example

Adaptive.addQueryMinMax('custom', 200, 400);

addQueryExpression(id: String, query: String)

Parameters:

  • id (String): Identifier for the query.
  • query (String): Media query string.

Returns: Void
Description: Registers a custom expression query.

Example

Adaptive.addQueryExpression('custom', 'screen and (max-width: 500em)');

if(breakdownId: String, callback: Function | Array): Object

Parameters:

  • breakdownId (String): Identifier like "tablet" or "mobile", etc.
  • callback (Function | Array): Function/Method or Array with object and property to set.

Returns: Object: Proxy
Description: Executes callback based on the media query match.

Example

Adaptive.if('mobile', [object, 'propertyId']) || Adaptive.if('mobile', () => {});

init()

Returns: Void
Description: Initializes or re-initializes after the DOM has loaded.

Example

Adaptive.init();

useVue(Vue: Object, hybrid: Boolean)

Parameters:

  • Vue (Object): Vue instance.
  • hybrid (Boolean): Allow support for both static and dynamic elements.

Returns: Vue
Description: Configures the library for use with Vue.

Example

Adaptive.useVue(VueInstance, true);

useReact(React: Object, hybrid: Boolean)

Parameters:

  • React (Object): React instance.
  • hybrid (Boolean): Allow support for both static and dynamic elements.

Returns: Void
Description: Configures the library for use with React.

Example

Adaptive.useReact(ReactInstance, true);

-- Teleport

Teleports an element temporarily or permanently [before,after,to] to a given target. Important because the order, position and hierarchy can be adapt better to the desired layout or device.

| prop | use | description | | ---------- | -------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------- | | before | teleport.mobile{before:target} --or-- teleport.mobile.before(target) | Teleports the element before the position of the target element (No merging. Good for cases where both should not inherit nor interfere with each other) | | after | teleport.mobile{after:target} --or-- teleport.mobile.after(target) | Teleports the element after the position of the target element (No merging. Good for cases where both should not inherit nor interfere with each other) | | to | teleport.mobile{to:target} --or-- teleport.mobile.to(target) | Teleports and appends the element to the target element (important if the element needs to inherit properties from the parent (target) element or be part of it) |

--

Consider the following graphic:

Screenshot_32

When using with Vue, there is no need to wrap it into custom tags (like Vue does), it uses the directive style v-teleport-to

-- Add/Remove Classes

Too many classes or rule-sets that override other rule-sets can be complex or hard. So rather than creating single use classes, this could be done like this example:

  • Need to change the class only at mobiles and desktop, others should be the base class:

  • <div class="bg-warning" data-adaptive="addClass.mobile|desktop(bg-info,w-100,p-5); removeClass.mobile|desktop('bg-warning')>...</div>

  • The above will result like:

  • Mobile and desktop devices:

  • <div class="bg-info,w-100,p-5"...>...</div>

  • Other devices

  • <div class="bg-warning"...>...</div>

-- Add/Remove Inline style

One off cases where it needs tweaks per breakdown. For instance:

  • On mobile devices the element needs 2px left margin
  • <div ... data-adaptive="addStyle.mobile(margin-left:2px;)>...</div>
  • The above will result in mobiles like:
    • <div style="margin-left:2px;"...>...</div>

-- Execute js functions at breakdows



// Stand alone or React
$adaptive.if('tablet', function () {
    // code
});

// OR
$adaptive.if('tablet', function () {
    // code
}).else(() => { });

// OR when it needs to execute only once
$adaptive.if('tablet', function () {
    // code
}).onlyOnce();

// OR
// For Vue
this.Adaptive.if('tablet', [this, 'ThisCustomMethod']);
// It will call a method "ThisCustomMethod"
data.tablet(reactive | function)


//OR
//Vue
this.Adaptive.if('tablet', function () {
    // code...
});

//OR
//Vue
this.Adaptive.if('mobile', function () {
    // code..
}).else(function () {
    //code..
});

// OR
//Vue callback
this.Adaptive.if('tablet', this.changeText);

-- Add custom queries

//Optional | Add custom media query (min px, max px) settings (min max)
Adaptive.addQueryMinMax('kitty', 900, 1400);

// Optional | Add a custom media query expression (it accepts any valid media query)
Adaptive.addQueryExpression('doggy', '(min-width: 900px)');

or directly in the element

<!-- Example of using custom queries -->
<div
    id="customQueries"
    data-adaptive="addClass[(min-width: 900px) and (max-width: 1599px)](custom-class-at-custom-breakdown)"
>
    Custom Expressions via string like Directive
</div>

--Out of the box present breakdowns

screens = {
    320: [1, 379],
    480: [380, 519],
    520: [520, 599] /* up to : mobiles */,
    600: [600, 699] /* up to : mid-size-tables */,
    700: [700, 799] /* up to : tablets / ipad */,
    800: [800, 919] /* transition in between tablets and desktop */,
    920: [920, 999] /* from here on for desktops */,
    1000: [1000, 1199],
    1200: [1200, 1439],
    1440: [1440, 1599],
    1600: [1600, 1700],
};

/**
 * break the 3 major device types
 */
devices = {
    mobile: [1, 599] /* Actual phones */,
    tablet: [600, 799] /* tablets in portrait or below */,
    'odd-device': [800, 1023] /* small Laptops and Ipads in landscape */,
    desktop: [1024, 1440] /* Most common resolutions below 1920 */,
};

/**
 * other non standard
 */
broadMediaQueries = {
    'non-desktop': [100, 1024],
    fullscreen: [1441, 6000] /* Large monitos and fullscreen in 1920 res */,
};

-- Easy sintax

  • Dot notation
<!-- Teleporting only at mobile and desktop, all others will be set back to original position -->
<div data-adaptive="teleport.desktop|mobile.before(#hello)">
    <span style="background:#1f2252; color: white;"
        >Static element that will teleport at "desktop or mobile" right above "static hello"</span
    >
</div>
  • Object like
<div
    data-adaptive="{'teleport':{
	'tablet': {
	'to':'.lazy'
}
}}"
>
    <span>Static element that will teleport at "tablet" size to a Vue driven element when visible</span>
</div>
  • Array like
<div data-teleport="[after,'#hello']">
    <span style="background:#083d39; color: white;"
        >2 Static element that will teleport "hello" with data attribute using an array like string</span
    >
</div>

-- All Adaptive.js Classes can be used as stand-alone

  • Adaptive: Main Adaptive.js
  • Observer: Easy register or unregister callback functions to be executed when any document node changes. (can be used stand alone or along with Adaptive.js)
  • Teleport: teleport elements to other locations (can be used stand alone or along with Adaptive.js)
  • ElementHelper: Element selector with some utility functions and Xpath support. (can be used stand alone or along with Adaptive.js)

The full description

So what's the story or even why today? In many projects, even when trying to stick to the philosophy of using mostly CSS utility classes and media queries to create responsive and adaptive layouts, CSS sometimes falls short in some areas, and that's why some JS is needed to help along those bumps.

But wait! what are those limitations you speak of! kind sir...

The short version.. is edge cases, class management and hierarchy, operations that are not currently part of CSS (teleporting, adding inline style on the fly, adding/removing classes, execute JS code at specific breakpoinst, etc)

For instance, moving elements from one location to another is not possible with CSS (only if the element belongs directly to the same parent and is a flex item (flex order)). In cases like that, where an element has to move that far, that's when Adaptive.js has a "teleport" feature that will move any element to specific targets at the specified Media query breakdown. Thus helping the layout not just being responsive but also adapting to the desired necessities. But that's not all Adaptive.js can help with, it can dynamically add and remove classes at different Media Breakdowns, or even to add 'inline' styles so that small tweaks are possible without adding endless classes that have only a single-one-time use.

Furthermore, Adaptive.js can add specific JS functions that will execute at a very specific screen size or media query, that is extremely helpful if you have JS functionality that should only execute at a very specific size or device!

Adaptive.js is not a CSS media query replacement nor should be the primary handler of responsive or adaptive applications (CSS should do most of it). On the contrary, it aims to enhance the ability to make layouts "adapt" better to different devices depending on preset or custom media queries via plain JavaScript or integrating with any of the cool frameworks out there (Vue, React, etc) and thus overcomimg the limitations of pure CSS or targeting complex bahaviors while avoiding WET and bloted code so that no JavaScript hacks or media functions need to be written all over, just using Adaptive will do all the magic.

(If you need some reference to understand responsive and adaptive head over here: https://www.geeksforgeeks.org/difference-between-responsive-design-and-adaptive-design/)

adaptive ex1

adaptive ex2

adaptive ex3

Bridging the Gaps:

Adaptive.js isn't merely a response to the shortcomings of CSS; it's a comprehensive solution to the challenges presented by many CSS frameworks. It abstracts the complexity, allowing developers to focus on creating exceptional user experiences rather than navigating through conflicting classes, puzzling rules or messy solutions.

Teleportation Without Boundaries:

With the "teleport" feature of Adaptive.js, moving elements across your design is no longer a chore confined to heavy frameworks like Vue or React. This feature enables effortless repositioning, eliminating the dependency on bulkier solutions.

Precision Functionality with Breakpoints:

Adaptive.js takes adaptability to a new level. Not only can you design with precision, but you can also dictate functionality based on screen size. This means executing specific JavaScript functions at exact CSS breakpoints. The design and functionality are now in sync, responding dynamically to user environments.

Dynamic Class and Attribute Management:

Adding or removing classes and attributes can often be a cumbersome task. With Adaptive.js, this process is streamlined. Whether it's tweaking a design element or enhancing functionality, Adaptive.js ensures it's done with ease and precision.

Tailored Designs with Custom CSS Queries:

While Adaptive.js has a preset breakdown for media queries, it also offers an API to craft custom CSS queries. This means designs that are not just responsive but truly bespoke, tailored to the unique requirements of each project.

Efficient, Clean, and Adaptable Code:

Redundant code is a developer's bane. Adaptive.js champions the DRY (Don't Repeat Yourself) principle, ensuring that every line of code serves a purpose, reducing clutter, and boosting efficiency.

Seamless Integration for a Cohesive Development Experience:

While Adaptive.js shines as a standalone tool, its real strength lies in its ability to play well with others. Its seamless integration with Vue and React ensures that you get the best of all worlds, enhancing and complementing without any friction.

Description

Inspiration used to build this: https://wicky.nillia.ms/enquire.js/ https://github.com/CyberAP/vue-component-media-queries https://developer.mozilla.org/en-US/docs/Web/API/Window/matchMedia https://developer.mozilla.org/en-US/docs/Web/API/MutationObserver https://vuejs.org/guide/built-ins/teleport.html

(This docs still under construction)