├── src ├── js │ ├── custom.js │ └── customMenu.js ├── images │ └── .gitkeep ├── scss │ ├── autoload │ │ ├── _autoload.scss │ │ └── _bootstrap.scss │ ├── layouts │ │ └── popUpSkin.scss │ ├── sections │ │ ├── _sections.scss │ │ ├── _header.scss │ │ └── _footer.scss │ ├── variables │ │ ├── _variables.scss │ │ ├── _type.scss │ │ ├── _colors.scss │ │ ├── _bs-overrides.scss │ │ └── _fonts.scss │ ├── components │ │ ├── _components.scss │ │ ├── _backgrounds.scss │ │ ├── _search.scss │ │ ├── _helpers.scss │ │ ├── _buttons.scss │ │ ├── _login.scss │ │ └── _nav.scss │ ├── style.scss │ ├── _base.scss │ ├── mixins │ │ └── _mixins.scss │ └── .scss-lint.yml └── fonts │ ├── OpenSans-Bold.woff2 │ ├── OpenSans-Italic.woff2 │ ├── OpenSans-Light.woff2 │ ├── OpenSans-Regular.woff2 │ ├── OpenSans-SemiBold.woff2 │ ├── OpenSans-BoldItalic.woff2 │ ├── OpenSans-ExtraBold.woff2 │ ├── OpenSans-LightItalic.woff2 │ ├── OpenSans-ExtraBoldItalic.woff2 │ └── OpenSans-SemiBoldItalic.woff2 ├── themeLicense.txt ├── .sonarcloud.properties ├── _config.yml ├── .gitignore ├── skin.doctype.xml ├── default.png ├── docs ├── _config.yml ├── Add-Custom-Theme-License.md ├── Add-Custom-Theme-Release-Notes.md ├── Change-Color-Scheme.md ├── Project-Setup.md ├── _data │ └── nav.yml ├── Dependencies.md ├── Work-With-Out-Of-Box-Theme.md ├── DevDependencies.md ├── Installation.md ├── README.md ├── _layouts │ └── default.html ├── SCSS.md └── Commands.md ├── thumbnail_default.png ├── menus ├── mobile │ ├── menudef.xml │ └── RazorMenu.cshtml └── desktop │ ├── menudef.xml │ └── RazorMenu.cshtml ├── containers └── none.ascx ├── popUpSkin.ascx ├── partials ├── _includes-popUpSkin.ascx ├── _registers-popUpSkin.ascx ├── _footer.ascx ├── _header.ascx ├── _includes.ascx └── _registers.ascx ├── project-details.json ├── koi.json ├── themeReleaseNotes.txt ├── .github ├── ISSUE_TEMPLATE │ ├── help-and-other-questions.md │ ├── feature-request.md │ ├── bug-report.md │ └── documentation-change-request.md ├── workflows │ └── build.yml ├── FUNDING.yml └── PULL_REQUEST_TEMPLATE.md ├── default.ascx ├── package.json ├── CODE_OF_CONDUCT.md ├── manifest.dnn ├── README.md ├── gulpfile.js └── LICENSE /src/js/custom.js: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /themeLicense.txt: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/images/.gitkeep: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /.sonarcloud.properties: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /_config.yml: -------------------------------------------------------------------------------- 1 | theme: jekyll-theme-architect -------------------------------------------------------------------------------- /src/scss/autoload/_autoload.scss: -------------------------------------------------------------------------------- 1 | @import 'bootstrap'; -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/ 2 | build/ 3 | dist/ 4 | yarn-error.log 5 | -------------------------------------------------------------------------------- /src/scss/autoload/_bootstrap.scss: -------------------------------------------------------------------------------- 1 | @import 'bootstrap/scss/bootstrap'; -------------------------------------------------------------------------------- /src/scss/layouts/popUpSkin.scss: -------------------------------------------------------------------------------- 1 | html { 2 | height: auto; 3 | } -------------------------------------------------------------------------------- /skin.doctype.xml: -------------------------------------------------------------------------------- 1 | ]]> -------------------------------------------------------------------------------- /src/scss/sections/_sections.scss: -------------------------------------------------------------------------------- 1 | @import 'header'; 2 | @import 'footer'; -------------------------------------------------------------------------------- /default.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/default.png -------------------------------------------------------------------------------- /docs/_config.yml: -------------------------------------------------------------------------------- 1 | theme: jekyll-theme-architect 2 | plugins: 3 | - jekyll-sitemap 4 | -------------------------------------------------------------------------------- /src/scss/variables/_variables.scss: -------------------------------------------------------------------------------- 1 | @import 'colors'; 2 | @import 'fonts'; 3 | @import 'type'; -------------------------------------------------------------------------------- /thumbnail_default.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/thumbnail_default.png -------------------------------------------------------------------------------- /src/fonts/OpenSans-Bold.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-Bold.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-Italic.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-Italic.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-Light.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-Light.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-Regular.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-Regular.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-SemiBold.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-SemiBold.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-BoldItalic.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-BoldItalic.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-ExtraBold.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-ExtraBold.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-LightItalic.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-LightItalic.woff2 -------------------------------------------------------------------------------- /menus/mobile/menudef.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /menus/desktop/menudef.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /src/fonts/OpenSans-ExtraBoldItalic.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-ExtraBoldItalic.woff2 -------------------------------------------------------------------------------- /src/fonts/OpenSans-SemiBoldItalic.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nvisionative/nvQuickTheme/HEAD/src/fonts/OpenSans-SemiBoldItalic.woff2 -------------------------------------------------------------------------------- /containers/none.ascx: -------------------------------------------------------------------------------- 1 | <%@ Control AutoEventWireup="false" Inherits="DotNetNuke.UI.Containers.Container" %> 2 |
3 | -------------------------------------------------------------------------------- /popUpSkin.ascx: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 |
-------------------------------------------------------------------------------- /src/scss/components/_components.scss: -------------------------------------------------------------------------------- 1 | @import 'dnn'; 2 | @import 'login'; 3 | @import 'backgrounds'; 4 | @import 'buttons'; 5 | @import 'helpers'; 6 | @import 'nav'; 7 | @import 'search'; -------------------------------------------------------------------------------- /src/scss/variables/_type.scss: -------------------------------------------------------------------------------- 1 | $base-font-size: 16; 2 | $lineheight: 1.5; 3 | $base-line-height: 24; 4 | $defaultlineheight: $base-font-size * 1.5; 5 | 6 | $open-sans: 'Open Sans', Helvetica, sans-serif; -------------------------------------------------------------------------------- /src/scss/components/_backgrounds.scss: -------------------------------------------------------------------------------- 1 | // bg classes from current color scheme 2 | @each $color, $value in $colors { 3 | .bg-#{$color}{ 4 | background-color:color($color) !important; 5 | } 6 | } -------------------------------------------------------------------------------- /src/scss/components/_search.scss: -------------------------------------------------------------------------------- 1 | ul.searchSkinObjectPreview > li > a.searchSkinObjectPreview_more { 2 | color: color('main-shade'); 3 | } 4 | #dnn_dnnSearch_ClassicSearch span { 5 | margin-right:.5rem; 6 | } -------------------------------------------------------------------------------- /src/scss/style.scss: -------------------------------------------------------------------------------- 1 | @import 'variables/bs-overrides'; 2 | @import 'autoload/autoload'; 3 | @import 'variables/variables'; 4 | @import 'mixins/mixins'; 5 | @import 'base'; 6 | @import 'components/components'; 7 | @import 'sections/sections'; -------------------------------------------------------------------------------- /partials/_includes-popUpSkin.ascx: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /src/scss/components/_helpers.scss: -------------------------------------------------------------------------------- 1 | // text classes from current color scheme 2 | @each $color, $value in $colors { 3 | .text-#{$color} { 4 | color: color($color) !important; 5 | } 6 | .border-#{$color} { 7 | border-color: color($color) !important; 8 | } 9 | } -------------------------------------------------------------------------------- /src/scss/sections/_header.scss: -------------------------------------------------------------------------------- 1 | header { 2 | .user-controls { 3 | display:flex; 4 | align-items:center; 5 | margin:0; 6 | margin-left:-1rem; 7 | margin-right:-1rem; 8 | li { 9 | padding:0 1rem; 10 | } 11 | } 12 | } -------------------------------------------------------------------------------- /project-details.json: -------------------------------------------------------------------------------- 1 | { 2 | "project": "nvQuickTheme", 3 | "version": "3.1.1", 4 | "author": "TK Sheppard & David Poindexter", 5 | "company": "nvisionative", 6 | "url": "www.nvquicktheme.com", 7 | "email": "support@nvisionative.com", 8 | "description": "A DNN Theme Dev Framework" 9 | } -------------------------------------------------------------------------------- /koi.json: -------------------------------------------------------------------------------- 1 | { 2 | "instructions": { 3 | "purpose": "This file is placed in the folder of DNN skins/themes. It tells the Koi system, what the primary CSS framework is for the themes in this folder.", 4 | "discoverMore": "https://connect-koi.net/" 5 | }, 6 | "default": { 7 | "cssFramework": "bs5" 8 | } 9 | } 10 | -------------------------------------------------------------------------------- /partials/_registers-popUpSkin.ascx: -------------------------------------------------------------------------------- 1 | <%@ Control Language="C#" AutoEventWireup="false" Explicit="True" Inherits="DotNetNuke.UI.Skins.Skin" %> 2 | <%@ Register TagPrefix="dnn" TagName="META" Src="~/Admin/Skins/Meta.ascx" %> 3 | <%@ Register TagPrefix="dnn" Namespace="DotNetNuke.Web.Client.ClientResourceManagement" Assembly="DotNetNuke.Web.Client" %> -------------------------------------------------------------------------------- /themeReleaseNotes.txt: -------------------------------------------------------------------------------- 1 |

________ Theme

2 |
A custom DNN theme designed and developed for the ________.
3 |
 
4 |

Release History

5 |
6 |
 
7 |
8 |

Version 1.0.0

9 |
    10 |
  • Initial Release
  • 11 |
12 |
-------------------------------------------------------------------------------- /docs/Add-Custom-Theme-License.md: -------------------------------------------------------------------------------- 1 | ## Add Custom Theme License 2 | As of Version 2.0.0 of **nvQuickTheme**, a new file is located in the root (`./`) folder of the project and is named `themeLicense.txt`. You can place a license within this file and the contents will automatically be displayed during the appropriate installation step within DNN using the project's output theme package. -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/help-and-other-questions.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Help and Other Questions 3 | about: Ask for help or others questions re. using the solution 4 | 5 | --- 6 | 7 | ## Please summarize your question in one sentence 8 | 9 | 10 | ## Give a more extended description 11 | 12 | 13 | ## Steps to reproduce (if needed) 14 | 15 | 16 | ## Other comments or remarks 17 | -------------------------------------------------------------------------------- /docs/Add-Custom-Theme-Release-Notes.md: -------------------------------------------------------------------------------- 1 | ## Add Custom Theme Release Notes 2 | As of Version 2.0.0 of **nvQuickTheme**, a new file is located in the root (`./`) folder of the project and is named `themeReleaseNotes.txt`. You can place release notes within this file and the contents will automatically be displayed during the appropriate installation step within DNN using the project's output theme package. -------------------------------------------------------------------------------- /src/scss/components/_buttons.scss: -------------------------------------------------------------------------------- 1 | // button iterations for each color 2 | @each $color, $value in $colors { 3 | .btn-#{$color} { 4 | @include button-variant($color,.8); 5 | } 6 | .btn-outline-#{$color} { 7 | @include button-outline-variant($color); 8 | } 9 | .btn-reverse-outline-#{$color} { 10 | @include button-reverse-outline-variant($color); 11 | } 12 | .btn-inverse-#{$color} { 13 | @include button-inverse-variant($color,.9); 14 | } 15 | } -------------------------------------------------------------------------------- /src/scss/sections/_footer.scss: -------------------------------------------------------------------------------- 1 | footer { 2 | .disclaimer { 3 | @include font-size(12); 4 | display:flex; 5 | margin:0; 6 | margin-left:-1rem; 7 | margin-right:-1rem; 8 | li { 9 | position:relative; 10 | padding:0 1rem; 11 | + li:before { 12 | content:''; 13 | position:absolute; 14 | top:0; 15 | left:0; 16 | bottom:0; 17 | width:1px; 18 | background-color:white; 19 | } 20 | } 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /default.ascx: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
9 | 10 |
11 | 12 |
13 |
14 |
15 |
16 |
17 | 18 |
19 | 20 |
21 | 22 | 23 | 24 | -------------------------------------------------------------------------------- /.github/workflows/build.yml: -------------------------------------------------------------------------------- 1 | name: Node.js CI 2 | 3 | on: 4 | push: 5 | branches: [ develop ] 6 | pull_request: 7 | branches: [ develop ] 8 | 9 | jobs: 10 | build: 11 | 12 | runs-on: ubuntu-latest 13 | 14 | strategy: 15 | matrix: 16 | node-version: [16.x] 17 | 18 | steps: 19 | - uses: actions/checkout@v3 20 | - name: Use Node.js ${{ matrix.node-version }} 21 | uses: actions/setup-node@v3 22 | with: 23 | node-version: ${{ matrix.node-version }} 24 | - name: Install dependencies 25 | run: yarn 26 | - name: Run gulp build 27 | run: gulp build 28 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/feature-request.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Feature Request 3 | about: Suggest an idea for this project 4 | 5 | --- 6 | 7 | ## Is your feature request related to a problem? 8 | **Please describe.** 9 | A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] 10 | 11 | ## Describe the solution you'd like 12 | A clear and concise description of what you want to happen. 13 | 14 | ## Describe alternatives you've considered 15 | A clear and concise description of any alternative solutions or features you've considered. 16 | 17 | ## Additional context 18 | Add any other context or screenshots about the feature request here. 19 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/bug-report.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Bug Report 3 | about: Create a report to help us improve 4 | 5 | --- 6 | 7 | ## Describe the bug 8 | A clear and concise description of what the bug is. 9 | 10 | ## To Reproduce 11 | Steps to reproduce the behavior: 12 | 1. Go to '...' 13 | 2. Click on '....' 14 | 3. Scroll down to '....' 15 | 4. See error 16 | 17 | ## Expected behavior 18 | A clear and concise description of what you expected to happen. 19 | 20 | ## Screenshots 21 | If applicable, add screenshots to help explain your problem. 22 | 23 | ## Errors 24 | Paste the error(s) related to this issue. 25 | 26 | ## Additional context 27 | Add any other context about the problem here. 28 | -------------------------------------------------------------------------------- /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | # These are supported funding model platforms 2 | 3 | github: [david-poindexter, nvisionative] # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] 4 | patreon: # Replace with a single Patreon username 5 | open_collective: # Replace with a single Open Collective username 6 | ko_fi: # Replace with a single Ko-fi username 7 | tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel 8 | community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry 9 | liberapay: # Replace with a single Liberapay username 10 | issuehunt: # Replace with a single IssueHunt username 11 | otechie: # Replace with a single Otechie username 12 | custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] 13 | -------------------------------------------------------------------------------- /docs/Change-Color-Scheme.md: -------------------------------------------------------------------------------- 1 | ### Color Scheme 2 | [![nvQuickTheme Video Series - Color Scheme](https://img.youtube.com/vi/o1XW3e8JKfw/0.jpg)](https://www.youtube.com/watch?v=o1XW3e8JKfw) 3 | 4 | ## Change Color Scheme 5 | Most websites are going to key off the colors for the brand being represented. Following industry best practices, there are typically three colors selected for the website main colors: primary, secondary and tertiary. These are defined within the `./src/scss/variables/_colors.scss` file. Updating these SCSS variable color definitions using HEX or RGB colors will change usage of these throughout the theme. 6 | 7 | There are also colors defined for variants of gray, normal font colors, borders and links. These are all easily modified within this one file. 8 | 9 | Additionally, base Bootstrap classes can be used for various colors within your custom theme. -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/documentation-change-request.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Documentation Change Request 3 | about: Suggest an improvement to project documentation 4 | 5 | --- 6 | 7 | ## What documentation page needs changing? 8 | Please reference the exact page on the documentation site that needs changing. 9 | 10 | ## What documentation section needs changing? 11 | Please reference the exact section within the page that needs changing. 12 | 13 | ## Proposed documentation change 14 | Provide a clear and concise description of the documentation change you wish to be made. 15 | 16 | ## Describe alternatives you've considered 17 | Provide a clear and concise description of any alternative documentation changes you considered. 18 | 19 | ## Additional context 20 | Provide any other context or screenshots that provide additional context for the proposed documentation changes. 21 | -------------------------------------------------------------------------------- /docs/Project-Setup.md: -------------------------------------------------------------------------------- 1 | ### Project Setup (Video) 2 | [![nvQuickTheme Video Series - Project Setup](https://img.youtube.com/vi/7UhpbUaeFQc/0.jpg)](https://www.youtube.com/watch?v=7UhpbUaeFQc) 3 | 4 | When you start a new theme project, you will need to edit **project-details.json** with your project information. 5 | 6 | ```javascript 7 | var project = '[project name]', 8 | author = '[author]', 9 | company = '[company]', 10 | url = '[website]', 11 | email = '[email]', 12 | description = '[short description]'; 13 | ``` 14 | 15 | Make sure you use html codes for any symbols used in these variables. After you've completed and saved the file, in the node command prompt within your root, you will need to run `gulp manifest`. This will take your information and populate the DNN manifest file with the information provided. -------------------------------------------------------------------------------- /partials/_footer.ascx: -------------------------------------------------------------------------------- 1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
    19 |
  • 20 |
  • 21 |
  • 22 |
23 |
24 |
25 |
26 |
27 |
28 | -------------------------------------------------------------------------------- /docs/_data/nav.yml: -------------------------------------------------------------------------------- 1 | menu: 2 | - title: Table of Contents 3 | subfolderitems: 4 | - page: Home 5 | url: /nvQuickTheme/ 6 | - page: Installation 7 | url: /nvQuickTheme/Installation.html 8 | - page: Project Setup 9 | url: /nvQuickTheme/Project-Setup.html 10 | - page: Commands 11 | url: /nvQuickTheme/Commands.html 12 | - title: Appendix 13 | subfolderitems: 14 | - page: Dependencies 15 | url: /nvQuickTheme/Dependencies.html 16 | - page: Dev Dependencies 17 | url: /nvQuickTheme/DevDependencies.html 18 | - page: SCSS 19 | url: /nvQuickTheme/SCSS.html 20 | - title: How To... 21 | subfolderitems: 22 | - page: Work With Out-Of-Box Theme 23 | url: /nvQuickTheme/Work-With-Out-Of-Box-Theme.html 24 | - page: Change Color Scheme 25 | url: /nvQuickTheme/Change-Color-Scheme.html 26 | - page: Add Custom Theme Release Notes 27 | url: /nvQuickTheme/Add-Custom-Theme-Release-Notes.html 28 | - page: Add Custom Theme License 29 | url: /nvQuickTheme/Add-Custom-Theme-License.html -------------------------------------------------------------------------------- /menus/desktop/RazorMenu.cshtml: -------------------------------------------------------------------------------- 1 | @using DotNetNuke.Web.DDRMenu; 2 | @using System.Dynamic; 3 | @inherits DotNetNuke.Web.Razor.DotNetNukeWebPage 4 | 5 | 8 | 9 | @helper RenderPages(IEnumerable pages) 10 | { 11 | if (!pages.Any()) 12 | { 13 | return; 14 | } 15 | 16 | foreach (var page in pages) 17 | { 18 | var hasChildren = page.HasChildren(); 19 | var attrTarget = !string.IsNullOrEmpty(page.Target) ? ("target=\"" + page.Target + "\"") :string.Empty; 20 | 21 |
  • 22 | @if (page.Enabled) 23 | { 24 | @page.Text 25 | } 26 | else 27 | { 28 | @page.Text 29 | } 30 | 31 | @if (hasChildren) 32 | { 33 |
      34 | @RenderPages(page.Children) 35 |
    36 | } 37 |
  • 38 | } 39 | } -------------------------------------------------------------------------------- /docs/Dependencies.md: -------------------------------------------------------------------------------- 1 | ### Bootstrap 2 | By far the largest dependency, Bootstrap, is the framework we chose to use. We encourage anyone unfamiliar with Bootstrap 4 to bookmark and reference their extensive documentation [here](https://getbootstrap.com/). 3 | 4 | Our **default.ascx**, **_header.ascx** and **_footer.ascx** partials show proper use of bootstrap and its layout classes. Bootstrap is robust, powerful, and has all the bells and whistles that most websites need. 5 | 6 | 7 | *** 8 | 9 | ### Font Awesome 10 | True to it's name, Font Awesome is just that, awesome. A font-based solution to icons allows for easy scaling without pixelation, and you're not left trying to manage lots of icon files. Font Awesome is easy to use, and has an icon for almost everything. You can find the list of icons [here](http://fontawesome.io/icons/). 11 | 12 | 13 | *** 14 | 15 | ### SlimMenu 16 | Where Bootstrap's menu options allow for simple menus, DNN usually requires multiple levels. SlimMenu is a super lightweight menu solution for desktop and mobile. 17 | 18 | 19 | *** 20 | 21 | ### Normalize.css 22 | A collection of CSS rules that make elements appear more consistently across all browser types. -------------------------------------------------------------------------------- /src/js/customMenu.js: -------------------------------------------------------------------------------- 1 | /*jshint esversion: 6 */ 2 | 3 | // add event listeners to expands 4 | const navExpand = [].slice.call(document.querySelectorAll('.nav-expand')); 5 | 6 | navExpand.forEach(item => { 7 | item.querySelector('.nav-expand-link').addEventListener('click', () => item.classList.add('active')); 8 | item.querySelector('.nav-back-link').addEventListener('click', () => item.classList.remove('active')); 9 | }); 10 | 11 | // setup and add overlay 12 | let overlay = document.createElement('div'); 13 | overlay.setAttribute('id', 'body-overlay'); 14 | document.body.appendChild(overlay).classList.add('d-none'); 15 | 16 | // target hamburger menu 17 | const responsiveMenu = document.getElementById('nav-mobile'); 18 | const bodyOverlay = document.getElementById('body-overlay'); 19 | 20 | responsiveMenu.addEventListener('click', function() { 21 | // toggle menu 22 | document.body.classList.toggle('nav-is-toggled'); 23 | bodyOverlay.classList.toggle('d-none'); 24 | }); 25 | 26 | // if overlay is clicked 27 | bodyOverlay.addEventListener('click', function(e) { 28 | // toggle menu and overlay 29 | document.body.classList.toggle('nav-is-toggled'); 30 | bodyOverlay.classList.toggle('d-none'); 31 | }); 32 | -------------------------------------------------------------------------------- /.github/PULL_REQUEST_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | 2 | ## Related to Issue 3 | Fixes # 4 | 5 | ## Description 6 | 7 | 8 | ## How Has This Been Tested? 9 | 10 | 11 | 12 | 13 | ## Screenshots (if appropriate): 14 | 15 | ## Types of changes 16 | 17 | - [ ] Bug fix (non-breaking change which fixes an issue) 18 | - [ ] New feature (non-breaking change which adds functionality) 19 | - [ ] Breaking change (fix or feature that would cause existing functionality to not work as expected) 20 | 21 | ## Checklist: 22 | 23 | 24 | - [ ] My code follows the code style of this project. 25 | - [ ] My change requires a change to the documentation. 26 | - [ ] I have updated the documentation accordingly. 27 | -------------------------------------------------------------------------------- /docs/Work-With-Out-Of-Box-Theme.md: -------------------------------------------------------------------------------- 1 | ## DNN Install Package 2 | With each new release of **nvQuickTheme**, we create a **DNN Install Package** for the out-of-box theme that **nvQuickTheme** produces. This is a highly optimized theme that can be used by anyone. These install packages can be downloaded from [HERE](https://github.com/nvisionative/nvQuickTheme/releases). The install package (ZIP file) will follow the following naming convention: 3 | 4 | ``` 5 | nvQuickTheme__Install.zip 6 | ``` 7 | 8 | Once you have the ZIP file downloaded, you may install it into the desired DNN instance in the same manner you install other themes, via the **Extensions** page. 9 | 10 | ### Important Note: 11 | Since the purpose of **nvQuickTheme** is more for development of custom themes, you wouldn't want to attempt to tweak the **Install Package** in order to create your custom theme. Instead, you will want to: 12 | * [Get your local development environment up and running](https://github.com/nvisionative/nvQuickTheme/wiki/Installation) 13 | * [Setup your custom project](https://github.com/nvisionative/nvQuickTheme/wiki/Project-Setup) 14 | * [Utilize the commands that are built-in the nvQuickTheme framework](https://github.com/nvisionative/nvQuickTheme/wiki/Commands) 15 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "nvquicktheme", 3 | "version": "3.1.1", 4 | "description": "Barebones Bootstrap 5 DNN Theme", 5 | "main": "gulpfile.js", 6 | "author": "nvisionative", 7 | "license": "GPL-3.0", 8 | "private": true, 9 | "dependencies": { 10 | "@fortawesome/fontawesome-free": "^6.5.1", 11 | "bootstrap": "^5.2.2", 12 | "normalize.css": "^8.0.1" 13 | }, 14 | "devDependencies": { 15 | "browser-sync": "^2.29.3", 16 | "fancy-log": "^2.0.0", 17 | "gulp": "^4.0.2", 18 | "gulp-autoprefixer": "^8.0.0", 19 | "gulp-clean": "^0.4.0", 20 | "gulp-clean-css": "^4.3.0", 21 | "gulp-imagemin": "^7.1.0", 22 | "gulp-jshint": "^2.1.0", 23 | "gulp-modernizr": "^4.0.3", 24 | "gulp-rename": "^2.0.0", 25 | "gulp-replace": "^1.1.3", 26 | "gulp-sass": "^5.1.0", 27 | "gulp-uglify-es": "^3.0.0", 28 | "gulp-webp": "^4.0.1", 29 | "gulp-zip": "^5.1.0", 30 | "imagemin-webp": "^6.0.0", 31 | "jshint": "^2.13.6", 32 | "sass": "^1.56.1" 33 | }, 34 | "browserslist": [ 35 | "> 1%", 36 | "ie >= 8", 37 | "edge >= 15", 38 | "ie_mob >= 10", 39 | "ff >= 45", 40 | "chrome >= 45", 41 | "safari >= 7", 42 | "opera >= 23", 43 | "ios >= 7", 44 | "android >= 4", 45 | "bb >= 10" 46 | ] 47 | } 48 | -------------------------------------------------------------------------------- /src/scss/variables/_colors.scss: -------------------------------------------------------------------------------- 1 | // Color Array 2 | $colors: ( 3 | "white":white, 4 | "light-shade":#e6e6e6, 5 | "light-accent":#e0e1eb, 6 | "main-shade":#ec3d46, 7 | "main-accent":#462a2b, 8 | "main-highlight":#00a5e1, 9 | "sub-shade":#005a7b, 10 | "sub-accent":#00a5e1, 11 | "sub-highlight":#48ceff, 12 | "dark-shade":#050621, 13 | "dark-highlight":#484858, 14 | "black":black, 15 | "dnnPrimaryAction": darkblue, // DNN Primary Button 16 | "dnnSecondaryAction": gainsboro, // DNN Secondary Button 17 | "dnnTertiaryAction": grey // DNN TertiaryButton 18 | ); 19 | 20 | // Convert Hex to RGB 21 | @function hex-to-rgb($hex) { 22 | @return red($hex), green($hex), blue($hex); 23 | } 24 | 25 | // Setup Color Array as CSS4 Variables 26 | :root { 27 | @each $name, $color in $colors { 28 | --color-#{$name}: #{$color}; 29 | } 30 | @each $name, $color in $colors { 31 | --color-#{$name}--rgb: #{hex-to-rgb($color)}; 32 | } 33 | } 34 | 35 | // Color Selector 36 | @function color($color-name, $rgb: false) { 37 | @if($rgb) { 38 | @return var(--color-#{$color-name}--rgb, map-get($colors, $color-name)); 39 | } @else { 40 | @return var(--color-#{$color-name}, map-get($colors, $color-name)); 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /partials/_header.ascx: -------------------------------------------------------------------------------- 1 |
    2 |
    3 |
    4 |
    5 |
    6 |
      7 |
    • 8 |
    • 9 |
    • 10 |
    • 11 |
    12 |
    13 |
    14 |
    15 |
    16 |
    17 |
    18 |
    19 |
    20 |
    22 |
    23 | 24 |
    25 |
    26 | 27 |
    28 |
    29 |
    30 |
    31 |
    -------------------------------------------------------------------------------- /src/scss/variables/_bs-overrides.scss: -------------------------------------------------------------------------------- 1 | /* Bootstrap Overrides */ 2 | 3 | // Bootstrap colors 4 | $theme-colors: ( 5 | 'primary':#ec3d46, 6 | 'secondary':#00a5e1, 7 | 'success':#28a745, 8 | 'info':#17a2b8, 9 | 'warning':#ffc107, 10 | 'danger':#dc3545, 11 | 'light':#f8f9fa, 12 | 'dark':#343a40 13 | ); 14 | 15 | // Grid breakpoints (changed to relative units to fix cross-platform inconsistencies) 16 | $grid-breakpoints: ( 17 | xs: 0, 18 | sm: 36rem, //576px 19 | md: 48rem, //768px 20 | lg: 62rem, //992px 21 | xl: 75rem, //1200px 22 | xxl: 87.5rem //1400px 23 | ); 24 | 25 | // Grid containers (changed to relative units to fix cross-platform inconsistencies) 26 | $container-max-widths: ( 27 | sm: 33.75rem, //540px 28 | md: 45rem, //720px 29 | lg: 60rem, //960px 30 | xl: 71.25rem, //1140px 31 | xxl: 82.5rem //1320px 32 | ); 33 | 34 | // Override Bootstrap spacers for more options 35 | $spacer:1rem; 36 | $spacers: ( 37 | 0: 0, 38 | half: calc($spacer / 2), 39 | 1: $spacer, 40 | 3half: ($spacer * 1.5), 41 | 2: ($spacer * 2), 42 | 3: ($spacer * 3), 43 | 4: ($spacer * 4), 44 | 5: ($spacer * 5), 45 | 6: ($spacer * 6), 46 | 7: ($spacer * 7), 47 | 8: ($spacer * 8) 48 | ); 49 | 50 | // Change gutter width 51 | $grid-gutter-width: 2rem; 52 | 53 | // Form Control changes 54 | $input-height:2rem; 55 | $input-color:#444; 56 | $input-font-size:13; 57 | $input-border-radius:2px; 58 | $input-padding-y:.25rem; 59 | $input-padding-x:.75rem; 60 | 61 | $form-check-input-margin-y:3px; 62 | 63 | $custom-file-height-inner:unset; 64 | $custom-file-line-height:1.61539; -------------------------------------------------------------------------------- /menus/mobile/RazorMenu.cshtml: -------------------------------------------------------------------------------- 1 | @using DotNetNuke.Web.DDRMenu; 2 | @using System.Dynamic; 3 | @inherits DotNetNuke.Web.Razor.DotNetNukeWebPage 4 | 5 | 6 | 7 | 12 | 13 | @helper RenderPages(IEnumerable pages) 14 | { 15 | if (!pages.Any()) 16 | { 17 | return; 18 | } 19 | 20 | foreach (var page in pages) 21 | { 22 | var hasChildren = page.HasChildren(); 23 | var pageDesc = Html.Raw(!string.IsNullOrEmpty(page.Description) ? ("
    " + @page.Description + "") :string.Empty); 24 | var attrTarget = !string.IsNullOrEmpty(page.Target) ? ("target=\"" + page.Target + "\"") :string.Empty; 25 | 26 | 49 | } 50 | } 51 | -------------------------------------------------------------------------------- /docs/DevDependencies.md: -------------------------------------------------------------------------------- 1 | ### Node 2 | From the website: "Node.js® is a JavaScript runtime built on Chrome's V8 JavaScript engine. Node.js uses an event-driven, non-blocking I/O model that makes it lightweight and efficient. Node.js' package ecosystem, npm, is the largest ecosystem of open source libraries in the world." 3 | 4 | 5 | *** 6 | 7 | ### Yarn 8 | Where npm set the groundwork for a package system, it falls short in many aspects. Yarn, the successor to Bower, helps manage packages with ease. Quick, secure, and reliable, Yarn is the front-runner for package management. 9 | 10 | 11 | *** 12 | 13 | ### Gulp 14 | Gulp is what we use to automate the painful stuff. error checking, compressing, minifying, zipping, anything you hate doing over and over, there's probably a way to automate it using Gulp and it's plethora of plugins. 15 | 16 | ### gulp-autoprefixer 17 | Forget about making sure you add the -webkit- prefix. This plugin automatically adds all the prefixes necessary for all the major browsers and versions. 18 | 19 | ### gulp-clean 20 | Allows for easy deletion of files. We use it to get rid of temporary files created when building packages. 21 | 22 | ### gulp-clean-css 23 | Allows for simple minification of CSS. 24 | 25 | ### gulp-imagemin 26 | Nicely compresses images as small as possible while maintaining quality. We don't use this in our main tasks, but is still available because we believe it is a necessary tool for a lot of theme creation. 27 | 28 | ### gulp-jshint 29 | Checks javascript for errors and lets you know what's wrong. 30 | 31 | ### gulp-notify 32 | Helps show messages in the node command line console. 33 | 34 | ### gulp-rename 35 | Used to rename files. 36 | 37 | ### gulp-replace 38 | Replaces string values within files. Not currently used, but will possibly be used in future releases of nvQuickTheme. 39 | 40 | ### gulp-sass 41 | This powerful plugin error checks, concatenates, and minifies scss. We use it for all the things. 42 | 43 | ### gulp-uglify 44 | A plugin that helps minify all the things. Since our sass plugin already does that for us, we use this to minify our javascript. 45 | 46 | ### gulp-zip 47 | Just like it sounds, this zips files. This is used in our theme packaging task to help create the resource files for DNN. -------------------------------------------------------------------------------- /docs/Installation.md: -------------------------------------------------------------------------------- 1 | ### Intro (Video) 2 | [![nvQuickTheme Video Series - Intro](https://img.youtube.com/vi/-w0qSTZfBUU/0.jpg)](https://www.youtube.com/watch?v=-w0qSTZfBUU) 3 | 4 | ### Install Node 5 | Node is a server framework that we use to keep track of all our dependencies and help run our scripts. Head over to [https://nodejs.org/en/](https://nodejs.org/en/) to find out how to install Node. 6 | 7 | Once you have Node installed, load up the Node command prompt. This is where you will be running all commands for the environment. 8 | 9 | 10 | *** 11 | 12 | ### Install Yarn 13 | Yarn is a package manager that we use to manage all our dependencies quickly and securely. 14 | 15 | Refer to the [yarn installation page](https://yarnpkg.com/en/docs/install) for details on your specific operating system. 16 | 17 | *** 18 | 19 | ### Install Gulp 20 | Gulp is our task script manager and executor. In order for us to run this on command line in any project we're doing, from the node command line, you will need to run: `npm install gulp -g`. This will install Gulp globally on your machine. Note that if you run several installations of nvQuickTheme, you will only need to run this command on the first install. 21 | 22 | *** 23 | 24 | ### Clone Repository 25 | The dev environment is setup to be run directly in a DNN instance under: `Portals\_default\Skins\[your theme name]`. Setup your empty Skin folder and clone our repo to that. 26 | 27 | Alternatively, if you are not able to run this in a DNN instance, or don't want to run it in an instance, you can set it up as is, wherever you would like. To test your theme, you would run the packaging command, then install your theme package on your testing instance. 28 | 29 | We recommend running it directly in a DNN instance for full ease-of-use. 30 | 31 | *** 32 | 33 | ### Install Dependencies 34 | With Yarn & Gulp installed, and the repo cloned, we can now use that package file. 35 | 36 | Run: `yarn` or `yarn install` 37 | 38 | This will read the package file and pull the necessary dependencies required for the environment. 39 | 40 | NOTE: Yarn will notify you that Bootstrap is missing some dependencies. Please ignore this warning. Bootstrap 4 includes all it's dependencies in the "bundle" version that we use. 41 | 42 | With all the dependencies installed and the repo cloned, your dev environment is ready to go! -------------------------------------------------------------------------------- /src/scss/_base.scss: -------------------------------------------------------------------------------- 1 | *, 2 | *:before, 3 | *:after { 4 | box-sizing: border-box; 5 | } 6 | 7 | 8 | 9 | // Base Styles 10 | //––––––––––––––––––––––––––––––––––––––––––––––––– 11 | 12 | body { 13 | font-size: 16px; 14 | font-weight: 200; 15 | line-height: 1.5; 16 | font-family: $open-sans; 17 | color: color('dark-shade'); 18 | background-color: #fff; 19 | -webkit-font-smoothing: antialiased; 20 | } 21 | 22 | h1, h2, h3, h4, h5 { 23 | font-family: $open-sans; 24 | } 25 | 26 | h1 { 27 | @include font-baseline(48, 52); 28 | } 29 | 30 | h2 { 31 | @include font-baseline(38, 48); 32 | } 33 | 34 | h3 { 35 | @include font-baseline(26, 36); 36 | } 37 | 38 | h4 { 39 | @include font-baseline(22, 28); 40 | } 41 | 42 | h5 { 43 | @include font-baseline(18, 24); 44 | } 45 | 46 | p { 47 | margin: 0 0 1.5rem; 48 | } 49 | 50 | ul, 51 | ol, 52 | pre, 53 | table, 54 | blockquote { 55 | margin-bottom: 1.5rem; 56 | } 57 | 58 | 59 | 60 | // Sanitation 61 | //––––––––––––––––––––––––––––––––––––––––––––––––– 62 | 63 | ul ul, 64 | ol ol, 65 | ul ol, 66 | ol ul { 67 | margin:0; 68 | } 69 | 70 | ul { 71 | list-style: disc; 72 | } 73 | 74 | ol { 75 | list-style: decimal; 76 | } 77 | 78 | b, 79 | strong, 80 | em, 81 | small, 82 | code { 83 | line-height: 1; 84 | } 85 | 86 | sup, 87 | sub { 88 | vertical-align: baseline; 89 | position: relative; 90 | top: -.4em; 91 | } 92 | 93 | sub { 94 | top: .4em; 95 | } 96 | 97 | 98 | 99 | // Links 100 | //––––––––––––––––––––––––––––––––––––––––––––––––– 101 | 102 | a { 103 | color: color('sub-highlight'); 104 | text-decoration:none; 105 | transition:.25s ease-in-out all; 106 | 107 | &:hover { 108 | color: color('light-shade'); 109 | text-decoration:none; 110 | } 111 | } 112 | header a, footer a { 113 | color: color('white'); 114 | font-weight:600; 115 | } 116 | 117 | 118 | 119 | // hr and pre 120 | //––––––––––––––––––––––––––––––––––––––––––––––––– 121 | 122 | hr { 123 | display:block; 124 | height:1px; 125 | width:25%; 126 | border:0; 127 | border-top:1px solid color('white'); 128 | margin-top:1.5rem; 129 | margin-bottom:1.5rem; 130 | padding: 0; 131 | } 132 | 133 | pre { 134 | background:color('sub-highlight'); 135 | padding:10px 15px; 136 | } -------------------------------------------------------------------------------- /src/scss/mixins/_mixins.scss: -------------------------------------------------------------------------------- 1 | @function strip-unit($num) { 2 | @return calc($num / ($num * 0 + 1)); 3 | } 4 | 5 | // rem font-size with px fallback 6 | // usage: @include font-size(16) 7 | @mixin font-size($sizeValue) { 8 | font-size: $sizeValue + px; 9 | font-size: calc($sizeValue / 16) + rem; 10 | } 11 | 12 | // vertical rhythm 13 | // usage: @include font-baseline(16,24); 14 | @mixin font-baseline($font-size, $lineheight){ 15 | font-size: $font-size + px; 16 | font-size: calc($font-size / $base-font-size) + rem; 17 | line-height: calc($lineheight / $font-size); 18 | margin: 0 0 calc((($lineheight / $font-size) * (1 / ($lineheight / $defaultlineheight))) * .5em ) 0; 19 | } 20 | 21 | // Shorthand for media queries at px width and lower 22 | @mixin mq-max-width( $width-in-px ) { 23 | @media only screen and (max-width: calc(($width-in-px / $base-font-size) * 1em )) { 24 | @content; 25 | } 26 | } 27 | 28 | // Shorthand for media queries at px width and higher 29 | @mixin mq-min-width( $width-in-px ) { 30 | @media only screen and (min-width: calc(($width-in-px / $base-font-size) * 1em )) { 31 | @content; 32 | } 33 | } 34 | 35 | // Regular solid buttons 36 | @mixin button-variant($color, $hover-alpha: 1) { 37 | $var-color: color($color); 38 | $var-rgb-color: color($color, true); 39 | color: white; 40 | background-color: $var-color; 41 | border-color: $var-color; 42 | &:hover, &:focus { 43 | color: white; 44 | background-color: rgba($var-rgb-color, $hover-alpha); 45 | border-color: rgba($var-rgb-color, $hover-alpha); 46 | } 47 | } 48 | 49 | // Outline buttons 50 | @mixin button-outline-variant($color) { 51 | $var-color: color($color); 52 | color: $var-color; 53 | background-color: transparent; 54 | border-color: $var-color; 55 | &:hover, &:focus { 56 | color: white; 57 | background-color: $var-color; 58 | border-color: $var-color; 59 | } 60 | } 61 | 62 | // Reverse outline to solid buttons 63 | @mixin button-reverse-outline-variant($color) { 64 | $var-color: color($color); 65 | color: white; 66 | background-color: $var-color; 67 | background-image: none; 68 | border-color: $var-color; 69 | &:hover, &:focus { 70 | color: $var-color; 71 | background-color: transparent; 72 | border-color: $var-color; 73 | } 74 | } 75 | 76 | // Inverse buttons to white background and colored text 77 | @mixin button-inverse-variant($color, $hover-alpha: 1) { 78 | $var-color: color($color); 79 | color: $var-color; 80 | background-color: color('white'); 81 | background-image: none; 82 | border-color: color('white'); 83 | &:hover, &:focus { 84 | color: $color; 85 | background-color: rgba(color('white', true), $hover-alpha); 86 | border-color: rgba(color('white', true), $hover-alpha); 87 | } 88 | } -------------------------------------------------------------------------------- /partials/_includes.ascx: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/scss/variables/_fonts.scss: -------------------------------------------------------------------------------- 1 | @font-face { 2 | font-family: 'Open Sans'; 3 | src: url('../fonts/OpenSans-Regular.woff2') format('woff2'), 4 | url('../fonts/OpenSans-Regular.woff') format('woff'); 5 | font-weight: normal; 6 | font-style: normal; 7 | font-display: swap; 8 | } 9 | 10 | @font-face { 11 | font-family: 'Open Sans'; 12 | src: url('../fonts/OpenSans-Italic.woff2') format('woff2'), 13 | url('../fonts/OpenSans-Italic.woff') format('woff'); 14 | font-weight: normal; 15 | font-style: italic; 16 | font-display: swap; 17 | } 18 | 19 | @font-face { 20 | font-family: 'Open Sans'; 21 | src: url('../fonts/OpenSans-Bold.woff2') format('woff2'), 22 | url('../fonts/OpenSans-Bold.woff') format('woff'); 23 | font-weight: bold; 24 | font-style: normal; 25 | font-display: swap; 26 | } 27 | 28 | @font-face { 29 | font-family: 'Open Sans'; 30 | src: url('../fonts/OpenSans-BoldItalic.woff2') format('woff2'), 31 | url('../fonts/OpenSans-BoldItalic.woff') format('woff'); 32 | font-weight: bold; 33 | font-style: italic; 34 | font-display: swap; 35 | } 36 | 37 | @font-face { 38 | font-family: 'Open Sans'; 39 | src: url('../fonts/OpenSans-ExtraBold.woff2') format('woff2'), 40 | url('../fonts/OpenSans-ExtraBold.woff') format('woff'); 41 | font-weight: 800; 42 | font-style: normal; 43 | font-display: swap; 44 | } 45 | 46 | @font-face { 47 | font-family: 'Open Sans'; 48 | src: url('../fonts/OpenSans-ExtraBoldItalic.woff2') format('woff2'), 49 | url('../fonts/OpenSans-ExtraBoldItalic.woff') format('woff'); 50 | font-weight: 800; 51 | font-style: italic; 52 | font-display: swap; 53 | } 54 | 55 | @font-face { 56 | font-family: 'Open Sans'; 57 | src: url('../fonts/OpenSans-Light.woff2') format('woff2'), 58 | url('../fonts/OpenSans-Light.woff') format('woff'); 59 | font-weight: 300; 60 | font-style: normal; 61 | font-display: swap; 62 | } 63 | 64 | @font-face { 65 | font-family: 'Open Sans'; 66 | src: url('../fonts/OpenSans-LightItalic.woff2') format('woff2'), 67 | url('../fonts/OpenSans-LightItalic.woff') format('woff'); 68 | font-weight: 300; 69 | font-style: italic; 70 | font-display: swap; 71 | } 72 | 73 | @font-face { 74 | font-family: 'Open Sans'; 75 | src: url('../fonts/OpenSans-SemiBold.woff2') format('woff2'), 76 | url('../fonts/OpenSans-SemiBold.woff') format('woff'); 77 | font-weight: 600; 78 | font-style: normal; 79 | font-display: swap; 80 | } 81 | 82 | @font-face { 83 | font-family: 'Open Sans'; 84 | src: url('../fonts/OpenSans-SemiBoldItalic.woff2') format('woff2'), 85 | url('../fonts/OpenSans-SemiBoldItalic.woff') format('woff'); 86 | font-weight: 600; 87 | font-style: italic; 88 | font-display: swap; 89 | } -------------------------------------------------------------------------------- /partials/_registers.ascx: -------------------------------------------------------------------------------- 1 | <%@ Control Language="C#" AutoEventWireup="true" Explicit="True" Inherits="DotNetNuke.UI.Skins.Skin" %> 2 | <%@ Register TagPrefix="dnn" TagName="BREADCRUMB" Src="~/Admin/Skins/Breadcrumb.ascx" %> 3 | <%@ Register TagPrefix="dnn" TagName="CONTROLPANEL" Src="~/Admin/Skins/ControlPanel.ascx" %> 4 | <%@ Register TagPrefix="dnn" TagName="COPYRIGHT" Src="~/Admin/Skins/Copyright.ascx" %> 5 | <%@ Register TagPrefix="dnn" TagName="CURRENTDATE" Src="~/Admin/Skins/CurrentDate.ascx" %> 6 | <%@ Register TagPrefix="dnn" TagName="DnnLink" Src="~/Admin/Skins/DnnLink.ascx" %> 7 | <%@ Register TagPrefix="dnn" TagName="DOTNETNUKE" Src="~/Admin/Skins/DotNetNuke.ascx" %> 8 | <%@ Register TagPrefix="dnn" TagName="Help" Src="~/Admin/Skins/Help.ascx" %> 9 | <%@ Register TagPrefix="dnn" TagName="HOSTNAME" Src="~/Admin/Skins/HostName.ascx" %> 10 | <%@ Register TagPrefix="dnn" TagName="JavaScriptLibraryInclude" Src="~/Admin/Skins/JavaScriptLibraryInclude.ascx" %> 11 | <%@ Register TagPrefix="dnn" TagName="JQUERY" Src="~/Admin/Skins/jQuery.ascx" %> 12 | <%@ Register TagPrefix="dnn" TagName="LANGUAGE" Src="~/Admin/Skins/Language.ascx" %> 13 | <%@ Register TagPrefix="dnn" TagName="LEFTMENU" Src="~/Admin/Skins/LeftMenu.ascx" %> 14 | <%@ Register TagPrefix="dnn" TagName="LINKS" Src="~/Admin/Skins/Links.ascx" %> 15 | <%@ Register TagPrefix="dnn" TagName="LINKTOFULLSITE" Src="~/Admin/Skins/LinkToFullSite.ascx" %> 16 | <%@ Register TagPrefix="dnn" TagName="LINKTOMOBILESITE" Src="~/Admin/Skins/LinkToMobileSite.ascx" %> 17 | <%@ Register TagPrefix="dnn" TagName="LOGIN" Src="~/Admin/Skins/Login.ascx" %> 18 | <%@ Register TagPrefix="dnn" TagName="LOGO" Src="~/Admin/Skins/Logo.ascx" %> 19 | <%@ Register TagPrefix="dnn" TagName="MENU" src="~/DesktopModules/DDRMenu/Menu.ascx" %> 20 | <%@ Register TagPrefix="dnn" TagName="META" Src="~/Admin/Skins/Meta.ascx" %> 21 | <%@ Register TagPrefix="dnn" TagName="MODULEMESSAGE" Src="~/Admin/Skins/ModuleMessage.ascx" %> 22 | <%@ Register TagPrefix="dnn" TagName="NAV" Src="~/Admin/Skins/Nav.ascx" %> 23 | <%@ Register TagPrefix="dnn" TagName="PRIVACY" Src="~/Admin/Skins/Privacy.ascx" %> 24 | <%@ Register TagPrefix="dnn" TagName="SEARCH" Src="~/Admin/Skins/Search.ascx" %> 25 | <%@ Register TagPrefix="dnn" TagName="STYLES" Src="~/Admin/Skins/Styles.ascx" %> 26 | <%@ Register TagPrefix="dnn" TagName="TAGS" Src="~/Admin/Skins/Tags.ascx" %> 27 | <%@ Register TagPrefix="dnn" TagName="TERMS" Src="~/Admin/Skins/Terms.ascx" %> 28 | <%@ Register TagPrefix="dnn" TagName="TEXT" Src="~/Admin/Skins/Text.ascx" %> 29 | <%@ Register TagPrefix="dnn" TagName="TOAST" Src="~/Admin/Skins/Toast.ascx" %> 30 | <%@ Register TagPrefix="dnn" TagName="TreeViewMenu" Src="~/Admin/Skins/TreeViewMenu.ascx" %> 31 | <%@ Register TagPrefix="dnn" TagName="USER" Src="~/Admin/Skins/user.ascx" %> 32 | <%@ Register TagPrefix="dnn" TagName="USERANDLOGIN" Src="~/Admin/Skins/UserAndLogin.ascx" %> 33 | <%@ Register TagPrefix="dnn" Namespace="DotNetNuke.Web.DDRMenu.TemplateEngine" Assembly="DotNetNuke.Web.DDRMenu" %> 34 | <%@ Register TagPrefix="dnn" Namespace="DotNetNuke.Web.Client.ClientResourceManagement" Assembly="DotNetNuke.Web.Client" %> 35 | -------------------------------------------------------------------------------- /CODE_OF_CONDUCT.md: -------------------------------------------------------------------------------- 1 | # Contributor Covenant Code of Conduct 2 | 3 | ## Our Pledge 4 | 5 | In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, gender identity and expression, level of experience, nationality, personal appearance, race, religion, or sexual identity and orientation. 6 | 7 | ## Our Standards 8 | 9 | Examples of behavior that contributes to creating a positive environment include: 10 | 11 | * Using welcoming and inclusive language 12 | * Being respectful of differing viewpoints and experiences 13 | * Gracefully accepting constructive criticism 14 | * Focusing on what is best for the community 15 | * Showing empathy towards other community members 16 | 17 | Examples of unacceptable behavior by participants include: 18 | 19 | * The use of sexualized language or imagery and unwelcome sexual attention or advances 20 | * Trolling, insulting/derogatory comments, and personal or political attacks 21 | * Public or private harassment 22 | * Publishing others' private information, such as a physical or electronic address, without explicit permission 23 | * Other conduct which could reasonably be considered inappropriate in a professional setting 24 | 25 | ## Our Responsibilities 26 | 27 | Project maintainers are responsible for clarifying the standards of acceptable behavior and are expected to take appropriate and fair corrective action in response to any instances of unacceptable behavior. 28 | 29 | Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, or to ban temporarily or permanently any contributor for other behaviors that they deem inappropriate, threatening, offensive, or harmful. 30 | 31 | ## Scope 32 | 33 | This Code of Conduct applies both within project spaces and in public spaces when an individual is representing the project or its community. Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. Representation of a project may be further defined and clarified by project maintainers. 34 | 35 | ## Enforcement 36 | 37 | Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting the project team at support@nvisionative.com. The project team will review and investigate all complaints, and will respond in a way that it deems appropriate to the circumstances. The project team is obligated to maintain confidentiality with regard to the reporter of an incident. Further details of specific enforcement policies may be posted separately. 38 | 39 | Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project's leadership. 40 | 41 | ## Attribution 42 | 43 | This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, available at [http://contributor-covenant.org/version/1/4][version] 44 | 45 | [homepage]: http://contributor-covenant.org 46 | [version]: http://contributor-covenant.org/version/1/4/ 47 | -------------------------------------------------------------------------------- /docs/README.md: -------------------------------------------------------------------------------- 1 | **nvQuickTheme** is more than just a great minimalist DNN (DotNetNuke) theme. It is a powerful theme building framework and developer workflow. This "starter solution" was made with developer efficiency and the ultimate DNN theme performance in mind. Most DNN themes in the marketplace today are loaded with bloat, which can make theming a pain, and for someone new to DNN, almost impossible. Our [documentation](https://nvisionative.github.io/nvQuickTheme/) will outline basic usage, modification, and advanced usage of this framework and developer workflow. 2 | 3 | Are you tired of spending countless hours trying to squeeze every ounce of performance out of your DNN themes, just so you can rank better on *Google PageSpeed Insights*? How are your scores? Have you broken the **80s for Desktop** or the **60s for Mobile**? You have probably discovered by now that it is very tough to accomplish this on a DNN site. 4 | 5 | Well, the search is over! Utilizing **nvQuickTheme**, you can rest assured that you are doing everything you possibly can within the context of the theme to achieve stellar scores. As a matter of fact, the [nvQuickTheme website](https://www.nvquicktheme.com), which of course is running nvQuickTheme, scores **100/100 (Desktop)** and **89/100 (Mobile)**. Yes, you read that correctly. If you don't believe us, please feel free to [analyze it yourself](https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fwww.nvquicktheme.com%2F&tab=desktop)! 6 | 7 | ## Background 8 | 9 | There have been many ways to develop DNN themes over the years. Some start from scratch using pure HTML with the [TOKEN] based model provided by DNN. Others, especially more hardcore developers, start with ASP.NET Web User Controls (instead of pure HTML). All end up using some shape, form, fashion or flavor of Cascading Style Sheets (CSS) and Javascript (JS). 10 | 11 | Some work with a manual workflow, editing flat files with their favorite code/text editor and wiring everything up "old school". Others use a fully Integrated Development Environment (IDE), like Visual Studio. Many even use the famous Christoc Visual Studio Templates for building out themes in a more structured way. Yet others develop their own unique workflows that best suit their particular use case. 12 | 13 | The [nvisionative](https://www.nvisionative.com) team has been working in the DNN ecosystem since its inception and has seen all sorts of approaches for custom and commercial themes alike. A large majority of these themes are bloated with all sorts of widgets, 3rd party components, bells, whistles, etc. Given the great depth of experience over the years, our team has identified so many ways to resolve issues in and improve DNN themes. From **best practices** to HUGE **performance enhancements**, there have been so many lessons learned. Now it is time give back to this great community that has provided so much to us. 14 | 15 | ## Video Series 16 | ### Intro 17 | [![nvQuickTheme Video Series - Intro](https://img.youtube.com/vi/-w0qSTZfBUU/0.jpg)](https://www.youtube.com/watch?v=-w0qSTZfBUU) 18 | 19 | ### Project Setup 20 | [![nvQuickTheme Video Series - Project Setup](https://img.youtube.com/vi/7UhpbUaeFQc/0.jpg)](https://www.youtube.com/watch?v=7UhpbUaeFQc) 21 | 22 | ### Color Scheme 23 | [![nvQuickTheme Video Series - Color Scheme](https://img.youtube.com/vi/o1XW3e8JKfw/0.jpg)](https://www.youtube.com/watch?v=o1XW3e8JKfw) -------------------------------------------------------------------------------- /src/scss/components/_login.scss: -------------------------------------------------------------------------------- 1 | .dnnLogin { 2 | display: flex; 3 | justify-content: center; 4 | width: 100%; 5 | 6 | > div { 7 | width:100%; 8 | padding:0; 9 | } 10 | 11 | .LoginPanel { 12 | width: 100%; 13 | padding: 0; 14 | 15 | .dnnFormItem { 16 | .dnnFormLabel { 17 | display:none; 18 | } 19 | 20 | &:nth-child(1), &:nth-child(2) { 21 | display: flex; 22 | flex-direction: column; 23 | 24 | .dnnFormLabel { 25 | display: block; 26 | float:left; 27 | &::after { 28 | clear:both; 29 | content: ''; 30 | } 31 | } 32 | 33 | label { 34 | font-weight: 700; 35 | font-size: 1.5rem; 36 | } 37 | 38 | input { 39 | min-width: 100%; 40 | font-size: 1.5rem; 41 | font-weight: 500; 42 | padding: .25em; 43 | } 44 | } 45 | 46 | &:nth-child(3) { 47 | width: 100%; 48 | display: flex; 49 | justify-content: center; 50 | margin-top: .5em; 51 | 52 | a { 53 | width: 50%; 54 | margin:0; 55 | 56 | &.dnnPrimaryAction { 57 | margin-right:.25em; 58 | } 59 | &.dnnSecondaryAction { 60 | margin-left:.25em; 61 | } 62 | } 63 | } 64 | 65 | .dnnLoginRememberMe { 66 | display: block; 67 | margin: 1em 0; 68 | 69 | img { 70 | margin: 0; 71 | padding: 0; 72 | } 73 | 74 | .dnnCheckbox { 75 | margin: 1em 0; 76 | display: inline-flex; 77 | vertical-align: middle; 78 | } 79 | 80 | label { 81 | margin-left: .5em; 82 | font-size: 1.25rem; 83 | font-weight: 550; 84 | } 85 | } 86 | 87 | &:last-child { 88 | width: 100%; 89 | 90 | .dnnLoginActions { 91 | width: 100%; 92 | 93 | ul { 94 | margin:0; 95 | padding:0; 96 | } 97 | li { 98 | width: 50%; 99 | margin: 0; 100 | 101 | &:nth-child(1) { 102 | padding-right: .25em; 103 | } 104 | 105 | &:nth-child(2) { 106 | padding-left: .25em; 107 | } 108 | } 109 | 110 | a { 111 | width: 100%; 112 | } 113 | } 114 | } 115 | } 116 | } 117 | } 118 | 119 | 120 | //Hides the #dnnHiddenLogin LOGIN theme object 121 | .hiddenLogin { 122 | bottom: 0; 123 | display: none; 124 | height: 30px; 125 | position: fixed; 126 | text-indent: -1000px; 127 | width: 30px; 128 | } -------------------------------------------------------------------------------- /docs/_layouts/default.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | {{ page.title | default: site.title | default: site.github.repository_name }} 5 | 6 | 7 | 8 | 9 | 10 | 11 | 14 | 15 | {% seo %} 16 | 17 | 18 | 19 |
    20 |
    21 | 22 |

    {{ site.title | default: site.github.repository_name }}

    23 |
    24 |

    {{ site.description | default: site.github.project_tagline }}

    25 | {% if site.github.is_project_page %} 26 | View project on GitHub 27 | {% endif %} 28 | {% if site.github.is_user_page %} 29 | Follow me on GitHub 30 | {% endif %} 31 |
    32 |
    33 | 34 |
    35 |
    36 |
    37 | nvQuickTheme 38 | 39 | {{ content }} 40 |
    41 | 42 | 68 | 71 |
    72 |
    73 | 74 | {% if site.google_analytics %} 75 | 83 | {% endif %} 84 | 85 | 86 | -------------------------------------------------------------------------------- /manifest.dnn: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | nvQuickTheme 5 | A DNN Theme Dev Framework 6 | MyIcon.png 7 | 8 | TK Sheppard & David Poindexter 9 | nvisionative 10 | www.nvquicktheme.com 11 | support@nvisionative.com 12 | 13 | 14 | 15 | 16 | 09.00.00 17 | 18 | 19 | 20 | 21 | Portals\_default\Skins\nvQuickTheme\ 22 | nvQuickTheme 23 | 24 | 25 | 26 | 27 | default.png 28 | 29 | 30 | 31 | thumbnail_default.png 32 | 33 | 34 | 35 | LICENSE 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | Portals\_default\Skins\nvQuickTheme\ 45 | 46 | else.zip 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | Portals\_default\Skins\nvQuickTheme\ 55 | 56 | dist.zip 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 | Portals\_default\Containers\nvQuickTheme\ 65 | 66 | cont.zip 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | web.config 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | 93 | 94 | 95 | 96 | 97 | 98 | 99 | 100 | 101 | 102 | -------------------------------------------------------------------------------- /docs/SCSS.md: -------------------------------------------------------------------------------- 1 | Our scss is located under `[root]/src/scss/`. For basic information on scss we recommend [this](https://css-tricks.com/sass-style-guide/) article. We will outline our scss organization methodology below. 2 | 3 | ### Adding custom .scss files 4 | Our organization is solid, but it won't always cover what you have. You will need to add your own .scss files for your projects, just try to keep with the organization! SCSS needs to be compiled, and to do so, there are instruction files setup to tell it how to compile, so when you add a new .scss file, you need to make sure you setup the instruction file to include your file. For almost all of them, they are named after the folder they are in. `/components/_components.scss` for instance: 5 | ```scss 6 | @import 'dnn'; 7 | @import 'buttons'; 8 | @import 'fonts'; 9 | @import 'forms'; 10 | @import 'grid'; 11 | @import 'helpers'; 12 | @import 'nav'; 13 | @import '[your new file]'; 14 | ``` 15 | 16 | ### Import Order 17 | It's important to keep in mind that order matters when compiling because that's the order it will be in the final css document. If you have scss that is dependent on previous scss, you will need to make sure that it's compiling in the correct order. The current order is as such: 18 | 19 | ```scss 20 | // needs variables first 21 | @import 'variables/variables'; 22 | @import 'breakpoints'; 23 | @import 'colors'; 24 | @import 'type'; 25 | 26 | // mixins will take variables but then needs to be before any mixin calls 27 | @import 'mixins/mixins'; 28 | 29 | // our baseline file 30 | @import 'base'; 31 | 32 | // components next in case page overrides on specific components are needed 33 | @import 'components/components'; 34 | @import 'dnn'; 35 | @import 'buttons'; 36 | @import 'fonts'; 37 | @import 'forms'; 38 | @import 'grid'; 39 | @import 'helpers'; 40 | @import 'nav'; 41 | 42 | // pages for template-wide specific styling 43 | @import 'pages/pages'; 44 | @import 'globals'; 45 | 46 | // and lastly sections with specific styling 47 | @import 'sections/sections'; 48 | @import 'header'; 49 | @import 'footer'; 50 | @import 'bannerpane'; 51 | ``` 52 | 53 | ### _base.scss 54 | This file is used for base element styling. Ex: 55 | ```scss 56 | // all nested lists have no margin 57 | ul ul, 58 | ol ol, 59 | ul ol, 60 | ol ul { 61 | margin:0; 62 | } 63 | ``` 64 | 65 | ### /variables/ 66 | Most scss variables should be located here. Any widely used variables created should be added here. Ex: 67 | ```scss 68 | $primary-color: rgb(236, 61, 70) !default; 69 | $secondary-color: rgb(0, 165, 225) !default; 70 | $tertiary-color: rgb(70, 42, 43) !default; 71 | ``` 72 | 73 | ### /mixins/_mixins.scss 74 | This file has all the scss mixins used, or any that you want to add. For clarity, mixins are shortcuts used within scss and should not be confused with helper css which are classes setup to use within the DOM. Ex: 75 | ```scss 76 | @mixin font-size($sizeValue) { 77 | font-size: $sizeValue + px; 78 | font-size: ($sizeValue / 16) + rem; 79 | } 80 | ``` 81 | Note: Don't add prefix mixins as gulp-autoprefixer adds all the prefixes needed upon compilation. 82 | 83 | ### /components/ 84 | Any website component styling should be done in this folder. We already have it broken down into several components, but you should add more if it makes sense, for instance, if you added an FAQ module with your own custom styling, you could add a `_faq.scss` to this folder. 85 | 86 | ### /pages/ 87 | This is where any page/template specific styling should be done. 88 | 89 | ### /sections/ 90 | Any section specific styling should be here. Ex: `_bannerpane.scss` was created here because it was possible this section would be reused on multiple templates. 91 | ```scss 92 | .bannerpane { 93 | background:$secondary-color; 94 | background:linear-gradient(135deg, darken($secondary-color, 20%) 0%, $secondary-color 50%, lighten($secondary-color, 20%) 100%); 95 | color:white; 96 | 97 | div[class*='col-'] { 98 | padding-top:6rem; 99 | padding-bottom:6rem; 100 | } 101 | } 102 | ``` -------------------------------------------------------------------------------- /docs/Commands.md: -------------------------------------------------------------------------------- 1 | Make sure you have navigated to your project root within your node command line. From this point you can use several commands. There are a few main dev commands, and then several smaller commands for more granular control. 2 | ## Main Dev Commands 3 | 4 | ### gulp init 5 | This command initializes all third-party dependencies utilized within **nvQuickTheme**. More specifically, it executes the following SubTask Commands in series (completes each task before the next): 6 | * **gulp fontsInit** 7 | * **gulp faFontsInit** 8 | * **gulp faCssInit** 9 | * **gulp slimMenuInit** 10 | * **gulp normalizeInit** 11 | * **gulp bsCssInit** 12 | * **gulp bsJsInit** 13 | 14 | ### gulp build 15 | This command cleans your `./dist/` folder, then error checks, concatenates, compiles and minifies all your JS and SCSS into the `./dist/` folder, as well as copies your containers to the correct folder. More specifically, it executes the following commands in series (completes each task before the next): 16 | * **gulp cleandist** 17 | * **gulp init** 18 | * **gulp styles** 19 | * **gulp scripts** 20 | * **gulp images** 21 | * **gulp containers** 22 | * **gulp manifest** 23 | 24 | ### gulp watch 25 | This sets your node instance to watch all changes of images, JS and SCSS in the `./src/` folder, as well as containers in the `./containers/` folder. Upon changes it will automatically run the **gulp build** command. 26 | 27 | **Note:** _This command will take over your node instance (in Command Prompt, PowerShell, Git Bash, VS Code Terminal, etc.). Therefore, you may want to start a new instance if you need to run other commands at the same time._ 28 | 29 | ### gulp packageTheme 30 | This command builds and packages your custom theme into a DNN theme installation package (ZIP file). It will place the ZIP file in the `./build/` folder using the following naming convention: `[project]_[version]_install.zip` 31 | 32 | 33 | *** 34 | 35 | ## SubTask Commands 36 | These commands are mostly used within the four Main Dev Commands, but can be used individually if desired. 37 | 38 | ### gulp manifest 39 | You should recognize this command from setting up your project. This command will update your manifest with the information provided in `project-details.json`. Anytime you need to update this information, like when changing the version number, you will need to run this command. This command is also executed as a part of the **gulp build** and **gulp package** commands. 40 | 41 | ### gulp scripts 42 | Error checks, concatenates, compiles, and minifies all JS in the `./src/js/` folder and distributes to the `./dist/js/` folder. 43 | 44 | ### gulp styles 45 | Error checks, concatenates, compiles, and minifies all SCSS in the `./src/scss/` folder and distributes to the `./dist/css/` folder. 46 | 47 | ### gulp images 48 | Compresses images in the `./src/images/` folder and distributes to the `./dist/images/` folder. 49 | 50 | ### gulp containers 51 | Copies containers to the correct folder within your DNN instance (assuming you are developing within a DNN instance). This translates to `../../Containers/[your theme project name]/`. 52 | 53 | 54 | *** 55 | 56 | ## Process Commands 57 | These commands are used within other commands and for other special situations. _We recommend use of these only for advanced users._ 58 | 59 | ### gulp cleandist 60 | Deletes contents of `./dist/` folder. This is particularly useful when assets (such as images) are no longer required and removed from `./src/` folder. It is the first task triggered during **gulp build**. 61 | 62 | ### gulp zipdist 63 | ZIPs contents of `./dist/` folder. Used to prepare for theme packaging. 64 | 65 | ### gulp zipcontainers 66 | ZIPs contents of `./containers/` folder. Used to prepare for theme packaging. 67 | 68 | ### gulp zipelse 69 | ZIPs contents of `./menus/` folder, `./partials/` folder, and all ASCX, XML, HTML and HTML files withinthe root folder (`./`). Used to prepare for theme packaging. 70 | 71 | ### gulp zippackage 72 | ZIPs all subset ZIP files and other pertinent project files into theme package installation file using the following naming convention: `[project]_[version]_install.zip` 73 | 74 | ### gulp cleantemp 75 | Deletes all temporary ZIP and project files from `./temp/` folder used in package tasks. 76 | -------------------------------------------------------------------------------- /src/scss/components/_nav.scss: -------------------------------------------------------------------------------- 1 | /* Main Navigation */ 2 | header { 3 | a[id*="dnnLOGO"] { 4 | display: block; 5 | padding: 1.125rem 0; 6 | 7 | img { 8 | height: 70px; 9 | max-height: 100%; 10 | width: auto; 11 | } 12 | 13 | @include mq-max-width(400) { 14 | img { 15 | height: 50px; 16 | } 17 | } 18 | } 19 | } 20 | 21 | /* Desktop Styling */ 22 | 23 | #nav-desktop { 24 | 25 | &, 26 | ul { 27 | display: flex; 28 | list-style: none; 29 | margin: 0; 30 | padding: 0; 31 | } 32 | 33 | li { 34 | display: flex; 35 | } 36 | 37 | a { 38 | @include font-size(16); 39 | color: color('main-accent'); 40 | font-weight: 600; 41 | text-decoration: none; 42 | text-transform: uppercase; 43 | 44 | &:hover { 45 | color: color('black'); 46 | } 47 | } 48 | 49 | // First Level 50 | >li { 51 | position: relative; 52 | 53 | &:hover { 54 | >ul { 55 | display: flex; 56 | } 57 | } 58 | 59 | >a { 60 | display: flex; 61 | align-items: flex-end; 62 | padding: 1rem 1.5rem; 63 | } 64 | 65 | // Below First Level 66 | ul { 67 | flex-direction: column; 68 | } 69 | 70 | li { 71 | &.selected>a { 72 | background: color('main-shade'); 73 | color: color('white'); 74 | } 75 | 76 | &:not(.selected)>a:hover { 77 | background: color('light-accent'); 78 | } 79 | } 80 | 81 | // Second Level 82 | >ul { 83 | display: none; 84 | position: absolute; 85 | flex-direction: column; 86 | top: 100%; 87 | min-width: 250px; 88 | max-width: 100%; 89 | background: color('white'); 90 | box-shadow: 1px 1px 5px 1px rgba(color('black', true), 0.2); 91 | 92 | // Below Second Level 93 | a { 94 | padding: 0.75rem 1.25rem; 95 | } 96 | 97 | li { 98 | flex-direction: column; 99 | 100 | &:not(:first-child) { 101 | // border-top: 1px solid color('dark-highlight'); 102 | } 103 | 104 | // Below Third Level 105 | >ul { 106 | 107 | // border-top: 1px solid color('dark-highlight'); 108 | a { 109 | @include font-size(14); 110 | font-weight: 700; 111 | padding: .5rem .75rem .5rem 3rem; 112 | } 113 | } 114 | } 115 | } 116 | } 117 | } 118 | 119 | /* Hamburger Styling */ 120 | 121 | $nav-height: 168px; 122 | $nav-width: 320px; 123 | 124 | #body-overlay { 125 | position: fixed; 126 | width: 100vw; 127 | height: 100vh; 128 | top: 0; 129 | right: 0; 130 | bottom: 0; 131 | left: 0; 132 | z-index: 1; 133 | background: rgba(black, .6); 134 | } 135 | 136 | .hamburger { 137 | margin-left: auto; 138 | color: color('main-accent'); 139 | cursor: pointer; 140 | } 141 | 142 | .nav-is-toggled { 143 | .nav-drill { 144 | transform: translateX(0); 145 | } 146 | } 147 | 148 | .nav { 149 | &-drill { 150 | display: flex; 151 | position: fixed; 152 | z-index: 100; 153 | top: 0; 154 | right: 0; 155 | width: $nav-width; 156 | max-width: 100%; 157 | height: calc(100vh - $nav-height); 158 | background-color: color('white'); 159 | overflow-y: auto; 160 | overflow-x: hidden; 161 | -webkit-overflow-scrolling: touch; 162 | transition: 0.45s; 163 | margin-top: $nav-height; 164 | transform: translateX(100%); 165 | 166 | ul { 167 | list-style: none; 168 | margin: 0; 169 | padding: 0; 170 | } 171 | } 172 | 173 | &-items { 174 | flex: 0 0 100%; 175 | } 176 | 177 | &-item { 178 | display: flex; 179 | justify-content: space-between; 180 | 181 | &:not(:last-child) { 182 | border-bottom: solid 1px color('dark-highlight'); 183 | } 184 | 185 | .nav-link { 186 | background-color: color('white'); 187 | color: color('main-accent'); 188 | } 189 | 190 | &.selected { 191 | background-color: color('main-shade'); 192 | 193 | >.nav-link { 194 | background-color: transparent; 195 | color: color('white'); 196 | } 197 | } 198 | 199 | .nav-back-link { 200 | display: flex; 201 | align-items: center; 202 | background: color('main-accent'); 203 | color: color('white'); 204 | 205 | i { 206 | font-size: 1.125rem; 207 | } 208 | } 209 | } 210 | 211 | &-link { 212 | display: block; 213 | flex: 1 1 auto; 214 | padding: 0.875rem 1rem; 215 | font-size: 1rem; 216 | line-height: 1.5em; 217 | font-weight: 500; 218 | 219 | .link-description { 220 | font-size: .875rem; 221 | font-weight: 400; 222 | } 223 | } 224 | 225 | &-expand { 226 | &-content { 227 | position: absolute; 228 | top: 0; 229 | left: 0; 230 | width: 100%; 231 | height: 100%; 232 | transform: translateX(100%); 233 | background-color: color('white'); 234 | transition: 0.3s; 235 | visibility: hidden; 236 | } 237 | 238 | &-link { 239 | flex-grow: 0; 240 | font-size: 1.125rem; 241 | display: flex; 242 | align-items: center; 243 | } 244 | 245 | &.active { 246 | >.nav-expand-content { 247 | transform: translateX(0); 248 | visibility: visible; 249 | } 250 | } 251 | } 252 | } -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | [![](https://www.nvquicktheme.com/Portals/0/nvQuickTheme_LOGO_final.png)](https://www.nvquicktheme.com) 2 | 3 | [![Node.js CI](https://github.com/nvisionative/nvQuickTheme/actions/workflows/build.yml/badge.svg)](https://github.com/nvisionative/nvQuickTheme/actions/workflows/build.yml) 4 | 5 | [![SonarCloud](https://sonarcloud.io/images/project_badges/sonarcloud-orange.svg)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) 6 | 7 | [![Lines of Code](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=ncloc)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=alert_status)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Duplicated Lines (%)](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=duplicated_lines_density)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Vulnerabilities](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=vulnerabilities)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Bugs](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=bugs)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Security Rating](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=security_rating)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Maintainability Rating](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=sqale_rating)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) [![Code Smells](https://sonarcloud.io/api/project_badges/measure?project=nvisionative_nvQuickTheme&metric=code_smells)](https://sonarcloud.io/summary/new_code?id=nvisionative_nvQuickTheme) 8 | 9 | **nvQuickTheme** is more than just a great minimalist DNN (DotNetNuke) theme. It is a powerful theme building framework and developer workflow. This "starter solution" was made with developer efficiency and the ultimate DNN theme performance in mind. Most DNN themes in the marketplace today are loaded with bloat, which can make theming a pain, and for someone new to DNN, almost impossible. Our [documentation](https://nvisionative.github.io/nvQuickTheme/) will outline basic usage, modification, and advanced usage of this framework and developer workflow. 10 | 11 | Are you tired of spending countless hours trying to squeeze every ounce of performance out of your DNN themes, just so you can rank better on *Google PageSpeed Insights*? How are your scores? Have you broken the **80s for Desktop** or the **60s for Mobile**? You have probably discovered by now that it is very tough to accomplish this on a DNN site. 12 | 13 | Well, the search is over! Utilizing **nvQuickTheme**, you can rest assured that you are doing everything you possibly can within the context of the theme to achieve stellar scores. As a matter of fact, the [nvQuickTheme website](http://www.nvquicktheme.com), which of course is running nvQuickTheme, scores **100/100 (Desktop)** and **89/100 (Mobile)**. Yes, you read that correctly. If you don't believe us, please feel free to [analyze it yourself](https://developers.google.com/speed/pagespeed/insights/?url=http%3A%2F%2Fwww.nvquicktheme.com%2F&tab=desktop)! 14 | 15 | ## Background 16 | 17 | There have been many ways to develop DNN themes over the years. Some start from scratch using pure HTML with the [TOKEN] based model provided by DNN. Others, especially more hardcore developers, start with ASP.NET Web User Controls (instead of pure HTML). All end up using some shape, form, fashion or flavor of Cascading Style Sheets (CSS) and Javascript (JS). 18 | 19 | Some work with a manual workflow, editing flat files with their favorite code/text editor and wiring everything up "old school". Others use a fully Integrated Development Environment (IDE), like Visual Studio. Many even use the famous Christoc Visual Studio Templates for building out themes in a more structured way. Yet others develop their own unique workflows that best suit their particular use case. 20 | 21 | The [nvisionative](http://www.nvisionative.com) team has been working in the DNN ecosystem since its inception and has seen all sorts of approaches for custom and commercial themes alike. A large majority of these themes are bloated with all sorts of widgets, 3rd party components, bells, whistles, etc. Given the great depth of experience over the years, our team has identified so many ways to resolve issues in and improve DNN themes. From **best practices** to HUGE **performance enhancements**, there have been so many lessons learned. Now it is time give back to this great community that has provided so much to us. 22 | 23 | ## Documentation 24 | [Learn How To Use nvQuickTheme](https://nvisionative.github.io/nvQuickTheme/) 25 | 26 | ## Video Series 27 | ### Intro 28 | [![nvQuickTheme Video Series - Intro](https://img.youtube.com/vi/-w0qSTZfBUU/0.jpg)](https://www.youtube.com/watch?v=-w0qSTZfBUU) 29 | 30 | ### Project Setup 31 | [![nvQuickTheme Video Series - Project Setup](https://img.youtube.com/vi/7UhpbUaeFQc/0.jpg)](https://www.youtube.com/watch?v=7UhpbUaeFQc) 32 | 33 | ### Color Scheme 34 | [![nvQuickTheme Video Series - Color Scheme](https://img.youtube.com/vi/o1XW3e8JKfw/0.jpg)](https://www.youtube.com/watch?v=o1XW3e8JKfw) 35 | 36 | --- 37 | [![Brought to the DNN community by nvisionative](http://www.nvquicktheme.com/Portals/0/broughtBy-nvisionative.png)](http://www.nvisionative.com) 38 | -------------------------------------------------------------------------------- /src/scss/.scss-lint.yml: -------------------------------------------------------------------------------- 1 | scss_files: "**/*.scss" 2 | 3 | # Default severity of all linters. 4 | severity: warning 5 | 6 | linters: 7 | BangFormat: 8 | enabled: true 9 | space_before_bang: true 10 | space_after_bang: false 11 | 12 | BemDepth: 13 | enabled: false 14 | max_elements: 1 15 | 16 | BorderZero: 17 | enabled: true 18 | convention: zero # or `none` 19 | 20 | ChainedClasses: 21 | enabled: false 22 | 23 | ColorKeyword: 24 | enabled: true 25 | 26 | ColorVariable: 27 | enabled: false 28 | 29 | Comment: 30 | enabled: true 31 | style: silent 32 | 33 | DebugStatement: 34 | enabled: true 35 | 36 | DeclarationOrder: 37 | enabled: true 38 | 39 | DisableLinterReason: 40 | enabled: false 41 | 42 | DuplicateProperty: 43 | enabled: true 44 | 45 | ElsePlacement: 46 | enabled: true 47 | style: same_line # or 'new_line' 48 | 49 | EmptyLineBetweenBlocks: 50 | enabled: true 51 | ignore_single_line_blocks: true 52 | 53 | EmptyRule: 54 | enabled: true 55 | 56 | ExtendDirective: 57 | enabled: false 58 | 59 | FinalNewline: 60 | enabled: true 61 | present: true 62 | 63 | HexLength: 64 | enabled: true 65 | style: short # or 'long' 66 | 67 | HexNotation: 68 | enabled: true 69 | style: lowercase # or 'uppercase' 70 | 71 | HexValidation: 72 | enabled: true 73 | 74 | IdSelector: 75 | enabled: true 76 | 77 | ImportantRule: 78 | enabled: true 79 | 80 | ImportPath: 81 | enabled: true 82 | leading_underscore: false 83 | filename_extension: false 84 | 85 | Indentation: 86 | enabled: true 87 | allow_non_nested_indentation: false 88 | character: space # or 'tab' 89 | width: 4 90 | 91 | LeadingZero: 92 | enabled: true 93 | style: exclude_zero # or 'include_zero' 94 | 95 | MergeableSelector: 96 | enabled: true 97 | force_nesting: true 98 | 99 | NameFormat: 100 | enabled: true 101 | allow_leading_underscore: true 102 | convention: hyphenated_lowercase # or 'camel_case', or 'snake_case', or a regex pattern 103 | 104 | NestingDepth: 105 | enabled: true 106 | max_depth: 5 107 | ignore_parent_selectors: false 108 | 109 | PlaceholderInExtend: 110 | enabled: true 111 | 112 | PropertyCount: 113 | enabled: false 114 | include_nested: false 115 | max_properties: 10 116 | 117 | PropertySortOrder: 118 | enabled: false 119 | ignore_unspecified: false 120 | min_properties: 2 121 | separate_groups: false 122 | 123 | PropertySpelling: 124 | enabled: true 125 | extra_properties: [] 126 | disabled_properties: [] 127 | 128 | PropertyUnits: 129 | enabled: true 130 | global: [ 131 | 'ch', 'em', 'ex', 'rem', # Font-relative lengths 132 | 'cm', 'in', 'mm', 'pc', 'pt', 'px', 'q', # Absolute lengths 133 | 'vh', 'vw', 'vmin', 'vmax', # Viewport-percentage lengths 134 | 'deg', 'grad', 'rad', 'turn', # Angle 135 | 'ms', 's', # Duration 136 | 'Hz', 'kHz', # Frequency 137 | 'dpi', 'dpcm', 'dppx', # Resolution 138 | '%'] # Other 139 | properties: {} 140 | 141 | PseudoElement: 142 | enabled: false 143 | 144 | QualifyingElement: 145 | enabled: true 146 | allow_element_with_attribute: false 147 | allow_element_with_class: false 148 | allow_element_with_id: false 149 | 150 | SelectorDepth: 151 | enabled: true 152 | max_depth: 5 153 | 154 | SelectorFormat: 155 | enabled: true 156 | convention: hyphenated_lowercase # or 'strict_BEM', or 'hyphenated_BEM', or 'snake_case', or 'camel_case', or a regex pattern 157 | 158 | Shorthand: 159 | enabled: true 160 | allowed_shorthands: [1, 2, 3] 161 | 162 | SingleLinePerProperty: 163 | enabled: true 164 | allow_single_line_rule_sets: true 165 | 166 | SingleLinePerSelector: 167 | enabled: true 168 | 169 | SpaceAfterComma: 170 | enabled: false 171 | style: one_space # or 'no_space', or 'at_least_one_space' 172 | 173 | SpaceAfterPropertyColon: 174 | enabled: false 175 | style: one_space # or 'no_space', or 'at_least_one_space', or 'aligned' 176 | 177 | SpaceAfterPropertyName: 178 | enabled: false 179 | 180 | SpaceAfterVariableName: 181 | enabled: true 182 | 183 | SpaceAroundOperator: 184 | enabled: true 185 | style: one_space # or 'at_least_one_space', or 'no_space' 186 | 187 | SpaceBeforeBrace: 188 | enabled: false 189 | style: space # or 'new_line' 190 | allow_single_line_padding: false 191 | 192 | SpaceBetweenParens: 193 | enabled: false 194 | spaces: 0 195 | 196 | StringQuotes: 197 | enabled: true 198 | style: single_quotes # or double_quotes 199 | 200 | TrailingSemicolon: 201 | enabled: true 202 | 203 | TrailingWhitespace: 204 | enabled: true 205 | 206 | TrailingZero: 207 | enabled: false 208 | 209 | TransitionAll: 210 | enabled: false 211 | 212 | UnnecessaryMantissa: 213 | enabled: true 214 | 215 | UnnecessaryParentReference: 216 | enabled: true 217 | 218 | UrlFormat: 219 | enabled: true 220 | 221 | UrlQuotes: 222 | enabled: true 223 | 224 | VariableForProperty: 225 | enabled: false 226 | properties: [] 227 | 228 | VendorPrefix: 229 | enabled: true 230 | identifier_list: base 231 | additional_identifiers: [] 232 | excluded_identifiers: [] 233 | 234 | ZeroUnit: 235 | enabled: true 236 | -------------------------------------------------------------------------------- /gulpfile.js: -------------------------------------------------------------------------------- 1 | const bs = require('browser-sync').create(), 2 | gulp = require('gulp'), 3 | autoprefixer = require('gulp-autoprefixer'), 4 | jshint = require('gulp-jshint'), 5 | sass = require('gulp-sass')(require('sass')), 6 | imagemin = require('gulp-imagemin'), 7 | modernizr = require('gulp-modernizr'), 8 | imwebp = require('imagemin-webp'), 9 | webp = require('gulp-webp'), 10 | rename = require('gulp-rename'), 11 | uglify = require('gulp-uglify-es').default, 12 | log = require('fancy-log'), 13 | replace = require('gulp-replace'), 14 | zip = require('gulp-zip'), 15 | clean = require('gulp-clean'), 16 | cleanCSS = require('gulp-clean-css'), 17 | details = require('./project-details.json'), 18 | project = details.project, 19 | version = details.version, 20 | author = details.author, 21 | company = details.company, 22 | url = details.url, 23 | email = details.email, 24 | description = details.description; 25 | 26 | const paths = { 27 | projectdetails: { 28 | src: './project-details.json' 29 | }, 30 | fonts: { 31 | src: './src/fonts/*', 32 | dest: './dist/fonts/' 33 | }, 34 | faFonts: { 35 | src: [ 36 | './node_modules/@fortawesome/fontawesome-free/webfonts/fa-brands-400.*', 37 | './node_modules/@fortawesome/fontawesome-free/webfonts/fa-solid-900.*' 38 | ], 39 | dest: './dist/webfonts/' 40 | }, 41 | faCss: { 42 | src: [ 43 | './node_modules/@fortawesome/fontawesome-free/css/brands.min.css', 44 | './node_modules/@fortawesome/fontawesome-free/css/solid.min.css', 45 | './node_modules/@fortawesome/fontawesome-free/css/fontawesome.min.css', 46 | ], 47 | dest: './dist/css/' 48 | }, 49 | normalize: { 50 | src: './node_modules/normalize.css/normalize.css', 51 | dest: './dist/css/' 52 | }, 53 | bsJs: { 54 | src: './node_modules/bootstrap/dist/js/bootstrap.bundle.min.*', 55 | dest: './dist/js/' 56 | }, 57 | images: { 58 | src: './src/images/**/*.{jpg,jpeg,png,gif,svg,webp}', 59 | dest: './dist/images/' 60 | }, 61 | styles: { 62 | src: './src/scss/**/*.scss', 63 | dest: './dist/css/' 64 | }, 65 | scripts: { 66 | src: './src/js/*.js', 67 | dest: './dist/js/' 68 | }, 69 | containers: { 70 | src: './containers/*', 71 | dest: '../../Containers/'+project+'/' 72 | }, 73 | manifest: { 74 | src: './manifest.dnn', 75 | dest: './' 76 | }, 77 | zipdist: { 78 | src: 'dist/**/*', 79 | zipfile: 'dist.zip', 80 | dest: './temp/' 81 | }, 82 | zipcontainers: { 83 | src: './containers/**/*', 84 | zipfile: 'cont.zip', 85 | dest: './temp/' 86 | }, 87 | zipelse: { 88 | src: ['./menus/**/*', './partials/*', '*.{ascx,xml,html,htm}', 'koi.json'], 89 | zipfile: 'else.zip', 90 | dest: './temp/' 91 | }, 92 | zippackage: { 93 | src: ['./temp/*.zip','*.{dnn,png,jpg,txt}', 'LICENSE'], 94 | zipfile: project+'\_'+version+'\_install.zip', 95 | dest: './build/' 96 | }, 97 | cleantemp: { 98 | src: './temp/' 99 | }, 100 | cleandist: { 101 | src: './dist/' 102 | } 103 | }; 104 | 105 | 106 | /*------------------------------------------------------*/ 107 | /* INIT TASKS ------------------------------------------*/ 108 | /*------------------------------------------------------*/ 109 | // Copy fonts from src/fonts to dist/fonts 110 | function fontsInit() { 111 | let nSrc=0; 112 | return gulp.src(paths.fonts.src) 113 | .pipe(gulp.dest(paths.fonts.dest)) 114 | .on('data', function() { nSrc+=1; }) 115 | .on('end', function() { 116 | log(nSrc, 'font files distributed!'); 117 | }) 118 | } 119 | 120 | // Copy fontawesome-free fonts from node_modules to dist/fonts 121 | function faFontsInit() { 122 | let nSrc=0; 123 | return gulp.src(paths.faFonts.src) 124 | .pipe(gulp.dest(paths.faFonts.dest)) 125 | .on('data', function() { nSrc+=1; }) 126 | .on('end', function() { 127 | log(nSrc, 'FontAwesome files distributed!'); 128 | }) 129 | } 130 | 131 | // Copy fontawesome-free CSS from node_modules to dist/css/fontawesome-free 132 | function faCssInit() { 133 | let nSrc=0; 134 | return gulp.src(paths.faCss.src) 135 | .pipe(gulp.dest(paths.faCss.dest)) 136 | .on('data', function() { nSrc+=1; }) 137 | .on('end', function() { 138 | log(nSrc, 'CSS files distributed!'); 139 | }) 140 | } 141 | 142 | // Compile normalize.css from node_modules and copy to dist/js 143 | function normalizeInit() { 144 | let nSrc=0; 145 | return gulp.src(paths.normalize.src, { sourcemaps: true }) 146 | .pipe(sass({outputStyle: 'compressed'}).on('error', sass.logError)) 147 | .pipe(cleanCSS()) 148 | .pipe(rename({suffix: '.min'})) 149 | .pipe(autoprefixer()) 150 | .pipe(gulp.dest(paths.normalize.dest, { sourcemaps: '.' })) 151 | .on('data', function() { nSrc+=1; }) 152 | .on('end', function() { 153 | log(nSrc, 'normalize files distributed!'); 154 | }) 155 | } 156 | 157 | // Copy bootstrap JS from node_modules to dist/js 158 | function bsJsInit() { 159 | let nSrc=0; 160 | return gulp.src(paths.bsJs.src) 161 | .pipe(gulp.dest(paths.bsJs.dest)) 162 | .on('data', function() { nSrc+=1; }) 163 | .on('end', function() { 164 | log(nSrc, 'Bootstrap JS files distributed!'); 165 | }) 166 | } 167 | 168 | // modernizr Init 169 | function modernizrInit() { 170 | let nSrc=0; 171 | return gulp.src(paths.scripts.src) 172 | .pipe(modernizr({ 173 | 'options': [ 174 | 'setClasses' 175 | ], 176 | 'tests': [ 177 | 'webp' 178 | ] 179 | })) 180 | .pipe(uglify()) 181 | .pipe(rename({suffix: '-custom.min'})) 182 | .pipe(gulp.dest(paths.scripts.dest)) 183 | .on('data', function() { nSrc+=1; }) 184 | .on('end', function() { 185 | log(nSrc, 'modernizr files distributed!'); 186 | }) 187 | } 188 | 189 | /*------------------------------------------------------*/ 190 | /* END INIT TASKS --------------------------------------*/ 191 | /*------------------------------------------------------*/ 192 | 193 | 194 | /*------------------------------------------------------*/ 195 | /* IMAGE TASKS -----------------------------------------*/ 196 | /*------------------------------------------------------*/ 197 | // Optimize images and copy to dist/images 198 | function optimize() { 199 | let nSrc=0; 200 | return gulp.src(paths.images.src, {since: gulp.lastRun(images)}) 201 | .pipe(imagemin([ 202 | imagemin.gifsicle({interlaced: true}), 203 | imagemin.mozjpeg({quality: 75, progressive: true}), 204 | imagemin.optipng({optimizationLevel: 5}), 205 | imagemin.svgo({ 206 | plugins: [ 207 | {removeViewBox: true}, 208 | {cleanupIDs: false} 209 | ] 210 | }) 211 | ], { 212 | plugins: imwebp({quality:75}) 213 | } 214 | )) 215 | .pipe(gulp.dest(paths.images.dest)) 216 | .on('data', function() { nSrc+=1; }) 217 | .on('end', function() { 218 | log(nSrc, 'images optimized!'); 219 | }) 220 | } 221 | 222 | // Make WebP versions of all images 223 | function convert() { 224 | let nSrc=0; 225 | return gulp.src(paths.images.src, {since: gulp.lastRun(images)}) 226 | .pipe(webp()) 227 | .pipe(gulp.dest(paths.images.dest)) 228 | .on('data', function() { nSrc+=1; }) 229 | .on('end', function() { 230 | log(nSrc, 'webp files created!'); 231 | }) 232 | } 233 | 234 | /*------------------------------------------------------*/ 235 | /* END IMAGE TASKS -------------------------------------*/ 236 | /*------------------------------------------------------*/ 237 | 238 | 239 | /*------------------------------------------------------*/ 240 | /* STYLES TASKS ----------------------------------------*/ 241 | /*------------------------------------------------------*/ 242 | // Compile custom SCSS to CSS and copy to dist/css 243 | function styles() { 244 | return gulp.src(paths.styles.src, { sourcemaps: true }) 245 | .pipe(sass({includePaths: ['./node_modules']},{outputStyle: 'compressed'}).on('error', sass.logError)) 246 | .pipe(cleanCSS()) 247 | .pipe(rename({suffix: '.min'})) 248 | .pipe(autoprefixer()) 249 | .pipe(gulp.dest(paths.styles.dest, { sourcemaps: '.' })) 250 | .on('end', function() { 251 | log('SCSS compiled, minified, and distributed!'); 252 | }) 253 | } 254 | /*------------------------------------------------------*/ 255 | /* END STYLES TASKS ------------------------------------*/ 256 | /*------------------------------------------------------*/ 257 | 258 | 259 | /*------------------------------------------------------*/ 260 | /* SCRIPTS TASKS ---------------------------------------*/ 261 | /*------------------------------------------------------*/ 262 | // Compile custom JS and copy to dist/js 263 | function scripts() { 264 | return gulp.src(paths.scripts.src, { sourcemaps: true }) 265 | .pipe(jshint()) 266 | .pipe(uglify()) 267 | .pipe(rename({suffix: '.min'})) 268 | .pipe(gulp.dest(paths.scripts.dest, { sourcemaps: '.' })) 269 | .pipe(jshint.reporter('default')) 270 | .pipe(jshint.reporter('fail')) 271 | .on('end', function() { 272 | log('JS uglified and distributed!'); 273 | }) 274 | } 275 | /*------------------------------------------------------*/ 276 | /* END SCRIPTS TASKS -----------------------------------*/ 277 | /*------------------------------------------------------*/ 278 | 279 | 280 | /*------------------------------------------------------*/ 281 | /* DNN TASKS -------------------------------------------*/ 282 | /*------------------------------------------------------*/ 283 | // Copy containers to proper DNN theme containers folder 284 | function containers() { 285 | let nSrc=0; 286 | return gulp.src(paths.containers.src) 287 | .pipe(gulp.dest(paths.containers.dest)) 288 | .on('data', function() { nSrc+=1; }) 289 | .on('end', function() { 290 | log(nSrc, 'container files distributed!'); 291 | }) 292 | } 293 | 294 | // Update manifest.dnn 295 | function manifest() { 296 | return gulp.src(paths.manifest.src) 297 | .pipe(replace(/\ { 409 | return match.replace(/(http:\/\/|https:\/\/)[a-zA-Z0-9.-]+\//g, `//${req.headers.host}/`); 410 | } 411 | }, 412 | ], 413 | }); 414 | gulp.watch(paths.images.src, images).on('change', bs.reload); 415 | gulp.watch(paths.styles.src, styles).on('change', bs.reload); 416 | gulp.watch(paths.scripts.src, scripts).on('change', bs.reload); 417 | gulp.watch(paths.containers.src, containers).on('change', bs.reload); 418 | } 419 | 420 | // gulp watch 421 | function watch() { 422 | gulp.watch(paths.images.src, images); 423 | gulp.watch(paths.styles.src, styles); 424 | gulp.watch(paths.scripts.src, scripts); 425 | gulp.watch(paths.containers.src, containers); 426 | } 427 | 428 | // gulp images 429 | const images = gulp.series(optimize, convert); 430 | 431 | // gulp init 432 | const init = gulp.series(fontsInit, faFontsInit, faCssInit, normalizeInit, bsJsInit, modernizrInit); 433 | 434 | // gulp build 435 | const build = gulp.series(cleandist, init, styles, scripts, images, containers, manifest); 436 | 437 | // gulp packageTheme 438 | const packageTheme = gulp.series(build, ziptemp, zippackage, cleantemp); 439 | /*------------------------------------------------------*/ 440 | /* END DEV TASKS ---------------------------------------*/ 441 | /*------------------------------------------------------*/ 442 | 443 | 444 | /*------------------------------------------------------*/ 445 | /* EXPORT TASKS ----------------------------------------*/ 446 | /*------------------------------------------------------*/ 447 | // You can use CommonJS `exports` module notation to declare tasks 448 | exports.fontsInit = fontsInit; 449 | exports.faFontsInit = faFontsInit; 450 | exports.faCssInit = faCssInit; 451 | exports.normalizeInit = normalizeInit; 452 | exports.bsJsInit = bsJsInit; 453 | exports.convert = convert; 454 | exports.optimize = optimize; 455 | exports.images = images; 456 | exports.modernizrInit = modernizrInit; 457 | exports.styles = styles; 458 | exports.scripts = scripts; 459 | exports.containers = containers; 460 | exports.manifest = manifest; 461 | exports.cleandist = cleandist; 462 | exports.zipdist = zipdist; 463 | exports.zipcontainers = zipcontainers; 464 | exports.zipelse = zipelse; 465 | exports.ziptemp = ziptemp; 466 | exports.zippackage = zippackage; 467 | exports.cleantemp = cleantemp; 468 | exports.serve = serve; 469 | exports.watch = watch; 470 | exports.init = init; 471 | exports.build = build; 472 | exports.packageTheme = packageTheme; 473 | 474 | // Define default task that can be called by just running `gulp` from cli 475 | exports.default = build; 476 | /*------------------------------------------------------*/ 477 | /* END EXPORT TASKS ------------------------------------*/ 478 | /*------------------------------------------------------*/ 479 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | GNU GENERAL PUBLIC LICENSE 2 | ========================== 3 | 4 | Version 3, 29 June 2007 5 | 6 | Copyright © 2007 Free Software Foundation, Inc. <> 7 | 8 | Everyone is permitted to copy and distribute verbatim copies of this license 9 | document, but changing it is not allowed. 10 | 11 | ## Preamble 12 | 13 | The GNU General Public License is a free, copyleft license for software and other 14 | kinds of works. 15 | 16 | The licenses for most software and other practical works are designed to take away 17 | your freedom to share and change the works. By contrast, the GNU General Public 18 | License is intended to guarantee your freedom to share and change all versions of a 19 | program--to make sure it remains free software for all its users. We, the Free 20 | Software Foundation, use the GNU General Public License for most of our software; it 21 | applies also to any other work released this way by its authors. You can apply it to 22 | your programs, too. 23 | 24 | When we speak of free software, we are referring to freedom, not price. Our General 25 | Public Licenses are designed to make sure that you have the freedom to distribute 26 | copies of free software (and charge for them if you wish), that you receive source 27 | code or can get it if you want it, that you can change the software or use pieces of 28 | it in new free programs, and that you know you can do these things. 29 | 30 | To protect your rights, we need to prevent others from denying you these rights or 31 | asking you to surrender the rights. Therefore, you have certain responsibilities if 32 | you distribute copies of the software, or if you modify it: responsibilities to 33 | respect the freedom of others. 34 | 35 | For example, if you distribute copies of such a program, whether gratis or for a fee, 36 | you must pass on to the recipients the same freedoms that you received. You must make 37 | sure that they, too, receive or can get the source code. And you must show them these 38 | terms so they know their rights. 39 | 40 | Developers that use the GNU GPL protect your rights with two steps: (1) assert 41 | copyright on the software, and (2) offer you this License giving you legal permission 42 | to copy, distribute and/or modify it. 43 | 44 | For the developers' and authors' protection, the GPL clearly explains that there is 45 | no warranty for this free software. For both users' and authors' sake, the GPL 46 | requires that modified versions be marked as changed, so that their problems will not 47 | be attributed erroneously to authors of previous versions. 48 | 49 | Some devices are designed to deny users access to install or run modified versions of 50 | the software inside them, although the manufacturer can do so. This is fundamentally 51 | incompatible with the aim of protecting users' freedom to change the software. The 52 | systematic pattern of such abuse occurs in the area of products for individuals to 53 | use, which is precisely where it is most unacceptable. Therefore, we have designed 54 | this version of the GPL to prohibit the practice for those products. If such problems 55 | arise substantially in other domains, we stand ready to extend this provision to 56 | those domains in future versions of the GPL, as needed to protect the freedom of 57 | users. 58 | 59 | Finally, every program is threatened constantly by software patents. States should 60 | not allow patents to restrict development and use of software on general-purpose 61 | computers, but in those that do, we wish to avoid the special danger that patents 62 | applied to a free program could make it effectively proprietary. To prevent this, the 63 | GPL assures that patents cannot be used to render the program non-free. 64 | 65 | The precise terms and conditions for copying, distribution and modification follow. 66 | 67 | ## TERMS AND CONDITIONS 68 | 69 | ### 0. Definitions. 70 | 71 | “This License” refers to version 3 of the GNU General Public License. 72 | 73 | “Copyright” also means copyright-like laws that apply to other kinds of 74 | works, such as semiconductor masks. 75 | 76 | “The Program” refers to any copyrightable work licensed under this 77 | License. Each licensee is addressed as “you”. “Licensees” and 78 | “recipients” may be individuals or organizations. 79 | 80 | To “modify” a work means to copy from or adapt all or part of the work in 81 | a fashion requiring copyright permission, other than the making of an exact copy. The 82 | resulting work is called a “modified version” of the earlier work or a 83 | work “based on” the earlier work. 84 | 85 | A “covered work” means either the unmodified Program or a work based on 86 | the Program. 87 | 88 | To “propagate” a work means to do anything with it that, without 89 | permission, would make you directly or secondarily liable for infringement under 90 | applicable copyright law, except executing it on a computer or modifying a private 91 | copy. Propagation includes copying, distribution (with or without modification), 92 | making available to the public, and in some countries other activities as well. 93 | 94 | To “convey” a work means any kind of propagation that enables other 95 | parties to make or receive copies. Mere interaction with a user through a computer 96 | network, with no transfer of a copy, is not conveying. 97 | 98 | An interactive user interface displays “Appropriate Legal Notices” to the 99 | extent that it includes a convenient and prominently visible feature that (1) 100 | displays an appropriate copyright notice, and (2) tells the user that there is no 101 | warranty for the work (except to the extent that warranties are provided), that 102 | licensees may convey the work under this License, and how to view a copy of this 103 | License. If the interface presents a list of user commands or options, such as a 104 | menu, a prominent item in the list meets this criterion. 105 | 106 | ### 1. Source Code. 107 | 108 | The “source code” for a work means the preferred form of the work for 109 | making modifications to it. “Object code” means any non-source form of a 110 | work. 111 | 112 | A “Standard Interface” means an interface that either is an official 113 | standard defined by a recognized standards body, or, in the case of interfaces 114 | specified for a particular programming language, one that is widely used among 115 | developers working in that language. 116 | 117 | The “System Libraries” of an executable work include anything, other than 118 | the work as a whole, that (a) is included in the normal form of packaging a Major 119 | Component, but which is not part of that Major Component, and (b) serves only to 120 | enable use of the work with that Major Component, or to implement a Standard 121 | Interface for which an implementation is available to the public in source code form. 122 | A “Major Component”, in this context, means a major essential component 123 | (kernel, window system, and so on) of the specific operating system (if any) on which 124 | the executable work runs, or a compiler used to produce the work, or an object code 125 | interpreter used to run it. 126 | 127 | The “Corresponding Source” for a work in object code form means all the 128 | source code needed to generate, install, and (for an executable work) run the object 129 | code and to modify the work, including scripts to control those activities. However, 130 | it does not include the work's System Libraries, or general-purpose tools or 131 | generally available free programs which are used unmodified in performing those 132 | activities but which are not part of the work. For example, Corresponding Source 133 | includes interface definition files associated with source files for the work, and 134 | the source code for shared libraries and dynamically linked subprograms that the work 135 | is specifically designed to require, such as by intimate data communication or 136 | control flow between those subprograms and other parts of the work. 137 | 138 | The Corresponding Source need not include anything that users can regenerate 139 | automatically from other parts of the Corresponding Source. 140 | 141 | The Corresponding Source for a work in source code form is that same work. 142 | 143 | ### 2. Basic Permissions. 144 | 145 | All rights granted under this License are granted for the term of copyright on the 146 | Program, and are irrevocable provided the stated conditions are met. This License 147 | explicitly affirms your unlimited permission to run the unmodified Program. The 148 | output from running a covered work is covered by this License only if the output, 149 | given its content, constitutes a covered work. This License acknowledges your rights 150 | of fair use or other equivalent, as provided by copyright law. 151 | 152 | You may make, run and propagate covered works that you do not convey, without 153 | conditions so long as your license otherwise remains in force. You may convey covered 154 | works to others for the sole purpose of having them make modifications exclusively 155 | for you, or provide you with facilities for running those works, provided that you 156 | comply with the terms of this License in conveying all material for which you do not 157 | control copyright. Those thus making or running the covered works for you must do so 158 | exclusively on your behalf, under your direction and control, on terms that prohibit 159 | them from making any copies of your copyrighted material outside their relationship 160 | with you. 161 | 162 | Conveying under any other circumstances is permitted solely under the conditions 163 | stated below. Sublicensing is not allowed; section 10 makes it unnecessary. 164 | 165 | ### 3. Protecting Users' Legal Rights From Anti-Circumvention Law. 166 | 167 | No covered work shall be deemed part of an effective technological measure under any 168 | applicable law fulfilling obligations under article 11 of the WIPO copyright treaty 169 | adopted on 20 December 1996, or similar laws prohibiting or restricting circumvention 170 | of such measures. 171 | 172 | When you convey a covered work, you waive any legal power to forbid circumvention of 173 | technological measures to the extent such circumvention is effected by exercising 174 | rights under this License with respect to the covered work, and you disclaim any 175 | intention to limit operation or modification of the work as a means of enforcing, 176 | against the work's users, your or third parties' legal rights to forbid circumvention 177 | of technological measures. 178 | 179 | ### 4. Conveying Verbatim Copies. 180 | 181 | You may convey verbatim copies of the Program's source code as you receive it, in any 182 | medium, provided that you conspicuously and appropriately publish on each copy an 183 | appropriate copyright notice; keep intact all notices stating that this License and 184 | any non-permissive terms added in accord with section 7 apply to the code; keep 185 | intact all notices of the absence of any warranty; and give all recipients a copy of 186 | this License along with the Program. 187 | 188 | You may charge any price or no price for each copy that you convey, and you may offer 189 | support or warranty protection for a fee. 190 | 191 | ### 5. Conveying Modified Source Versions. 192 | 193 | You may convey a work based on the Program, or the modifications to produce it from 194 | the Program, in the form of source code under the terms of section 4, provided that 195 | you also meet all of these conditions: 196 | 197 | * **a)** The work must carry prominent notices stating that you modified it, and giving a 198 | relevant date. 199 | * **b)** The work must carry prominent notices stating that it is released under this 200 | License and any conditions added under section 7. This requirement modifies the 201 | requirement in section 4 to “keep intact all notices”. 202 | * **c)** You must license the entire work, as a whole, under this License to anyone who 203 | comes into possession of a copy. This License will therefore apply, along with any 204 | applicable section 7 additional terms, to the whole of the work, and all its parts, 205 | regardless of how they are packaged. This License gives no permission to license the 206 | work in any other way, but it does not invalidate such permission if you have 207 | separately received it. 208 | * **d)** If the work has interactive user interfaces, each must display Appropriate Legal 209 | Notices; however, if the Program has interactive interfaces that do not display 210 | Appropriate Legal Notices, your work need not make them do so. 211 | 212 | A compilation of a covered work with other separate and independent works, which are 213 | not by their nature extensions of the covered work, and which are not combined with 214 | it such as to form a larger program, in or on a volume of a storage or distribution 215 | medium, is called an “aggregate” if the compilation and its resulting 216 | copyright are not used to limit the access or legal rights of the compilation's users 217 | beyond what the individual works permit. Inclusion of a covered work in an aggregate 218 | does not cause this License to apply to the other parts of the aggregate. 219 | 220 | ### 6. Conveying Non-Source Forms. 221 | 222 | You may convey a covered work in object code form under the terms of sections 4 and 223 | 5, provided that you also convey the machine-readable Corresponding Source under the 224 | terms of this License, in one of these ways: 225 | 226 | * **a)** Convey the object code in, or embodied in, a physical product (including a 227 | physical distribution medium), accompanied by the Corresponding Source fixed on a 228 | durable physical medium customarily used for software interchange. 229 | * **b)** Convey the object code in, or embodied in, a physical product (including a 230 | physical distribution medium), accompanied by a written offer, valid for at least 231 | three years and valid for as long as you offer spare parts or customer support for 232 | that product model, to give anyone who possesses the object code either (1) a copy of 233 | the Corresponding Source for all the software in the product that is covered by this 234 | License, on a durable physical medium customarily used for software interchange, for 235 | a price no more than your reasonable cost of physically performing this conveying of 236 | source, or (2) access to copy the Corresponding Source from a network server at no 237 | charge. 238 | * **c)** Convey individual copies of the object code with a copy of the written offer to 239 | provide the Corresponding Source. This alternative is allowed only occasionally and 240 | noncommercially, and only if you received the object code with such an offer, in 241 | accord with subsection 6b. 242 | * **d)** Convey the object code by offering access from a designated place (gratis or for 243 | a charge), and offer equivalent access to the Corresponding Source in the same way 244 | through the same place at no further charge. You need not require recipients to copy 245 | the Corresponding Source along with the object code. If the place to copy the object 246 | code is a network server, the Corresponding Source may be on a different server 247 | (operated by you or a third party) that supports equivalent copying facilities, 248 | provided you maintain clear directions next to the object code saying where to find 249 | the Corresponding Source. Regardless of what server hosts the Corresponding Source, 250 | you remain obligated to ensure that it is available for as long as needed to satisfy 251 | these requirements. 252 | * **e)** Convey the object code using peer-to-peer transmission, provided you inform 253 | other peers where the object code and Corresponding Source of the work are being 254 | offered to the general public at no charge under subsection 6d. 255 | 256 | A separable portion of the object code, whose source code is excluded from the 257 | Corresponding Source as a System Library, need not be included in conveying the 258 | object code work. 259 | 260 | A “User Product” is either (1) a “consumer product”, which 261 | means any tangible personal property which is normally used for personal, family, or 262 | household purposes, or (2) anything designed or sold for incorporation into a 263 | dwelling. In determining whether a product is a consumer product, doubtful cases 264 | shall be resolved in favor of coverage. For a particular product received by a 265 | particular user, “normally used” refers to a typical or common use of 266 | that class of product, regardless of the status of the particular user or of the way 267 | in which the particular user actually uses, or expects or is expected to use, the 268 | product. A product is a consumer product regardless of whether the product has 269 | substantial commercial, industrial or non-consumer uses, unless such uses represent 270 | the only significant mode of use of the product. 271 | 272 | “Installation Information” for a User Product means any methods, 273 | procedures, authorization keys, or other information required to install and execute 274 | modified versions of a covered work in that User Product from a modified version of 275 | its Corresponding Source. The information must suffice to ensure that the continued 276 | functioning of the modified object code is in no case prevented or interfered with 277 | solely because modification has been made. 278 | 279 | If you convey an object code work under this section in, or with, or specifically for 280 | use in, a User Product, and the conveying occurs as part of a transaction in which 281 | the right of possession and use of the User Product is transferred to the recipient 282 | in perpetuity or for a fixed term (regardless of how the transaction is 283 | characterized), the Corresponding Source conveyed under this section must be 284 | accompanied by the Installation Information. But this requirement does not apply if 285 | neither you nor any third party retains the ability to install modified object code 286 | on the User Product (for example, the work has been installed in ROM). 287 | 288 | The requirement to provide Installation Information does not include a requirement to 289 | continue to provide support service, warranty, or updates for a work that has been 290 | modified or installed by the recipient, or for the User Product in which it has been 291 | modified or installed. Access to a network may be denied when the modification itself 292 | materially and adversely affects the operation of the network or violates the rules 293 | and protocols for communication across the network. 294 | 295 | Corresponding Source conveyed, and Installation Information provided, in accord with 296 | this section must be in a format that is publicly documented (and with an 297 | implementation available to the public in source code form), and must require no 298 | special password or key for unpacking, reading or copying. 299 | 300 | ### 7. Additional Terms. 301 | 302 | “Additional permissions” are terms that supplement the terms of this 303 | License by making exceptions from one or more of its conditions. Additional 304 | permissions that are applicable to the entire Program shall be treated as though they 305 | were included in this License, to the extent that they are valid under applicable 306 | law. If additional permissions apply only to part of the Program, that part may be 307 | used separately under those permissions, but the entire Program remains governed by 308 | this License without regard to the additional permissions. 309 | 310 | When you convey a copy of a covered work, you may at your option remove any 311 | additional permissions from that copy, or from any part of it. (Additional 312 | permissions may be written to require their own removal in certain cases when you 313 | modify the work.) You may place additional permissions on material, added by you to a 314 | covered work, for which you have or can give appropriate copyright permission. 315 | 316 | Notwithstanding any other provision of this License, for material you add to a 317 | covered work, you may (if authorized by the copyright holders of that material) 318 | supplement the terms of this License with terms: 319 | 320 | * **a)** Disclaiming warranty or limiting liability differently from the terms of 321 | sections 15 and 16 of this License; or 322 | * **b)** Requiring preservation of specified reasonable legal notices or author 323 | attributions in that material or in the Appropriate Legal Notices displayed by works 324 | containing it; or 325 | * **c)** Prohibiting misrepresentation of the origin of that material, or requiring that 326 | modified versions of such material be marked in reasonable ways as different from the 327 | original version; or 328 | * **d)** Limiting the use for publicity purposes of names of licensors or authors of the 329 | material; or 330 | * **e)** Declining to grant rights under trademark law for use of some trade names, 331 | trademarks, or service marks; or 332 | * **f)** Requiring indemnification of licensors and authors of that material by anyone 333 | who conveys the material (or modified versions of it) with contractual assumptions of 334 | liability to the recipient, for any liability that these contractual assumptions 335 | directly impose on those licensors and authors. 336 | 337 | All other non-permissive additional terms are considered “further 338 | restrictions” within the meaning of section 10. If the Program as you received 339 | it, or any part of it, contains a notice stating that it is governed by this License 340 | along with a term that is a further restriction, you may remove that term. If a 341 | license document contains a further restriction but permits relicensing or conveying 342 | under this License, you may add to a covered work material governed by the terms of 343 | that license document, provided that the further restriction does not survive such 344 | relicensing or conveying. 345 | 346 | If you add terms to a covered work in accord with this section, you must place, in 347 | the relevant source files, a statement of the additional terms that apply to those 348 | files, or a notice indicating where to find the applicable terms. 349 | 350 | Additional terms, permissive or non-permissive, may be stated in the form of a 351 | separately written license, or stated as exceptions; the above requirements apply 352 | either way. 353 | 354 | ### 8. Termination. 355 | 356 | You may not propagate or modify a covered work except as expressly provided under 357 | this License. Any attempt otherwise to propagate or modify it is void, and will 358 | automatically terminate your rights under this License (including any patent licenses 359 | granted under the third paragraph of section 11). 360 | 361 | However, if you cease all violation of this License, then your license from a 362 | particular copyright holder is reinstated (a) provisionally, unless and until the 363 | copyright holder explicitly and finally terminates your license, and (b) permanently, 364 | if the copyright holder fails to notify you of the violation by some reasonable means 365 | prior to 60 days after the cessation. 366 | 367 | Moreover, your license from a particular copyright holder is reinstated permanently 368 | if the copyright holder notifies you of the violation by some reasonable means, this 369 | is the first time you have received notice of violation of this License (for any 370 | work) from that copyright holder, and you cure the violation prior to 30 days after 371 | your receipt of the notice. 372 | 373 | Termination of your rights under this section does not terminate the licenses of 374 | parties who have received copies or rights from you under this License. If your 375 | rights have been terminated and not permanently reinstated, you do not qualify to 376 | receive new licenses for the same material under section 10. 377 | 378 | ### 9. Acceptance Not Required for Having Copies. 379 | 380 | You are not required to accept this License in order to receive or run a copy of the 381 | Program. Ancillary propagation of a covered work occurring solely as a consequence of 382 | using peer-to-peer transmission to receive a copy likewise does not require 383 | acceptance. However, nothing other than this License grants you permission to 384 | propagate or modify any covered work. These actions infringe copyright if you do not 385 | accept this License. Therefore, by modifying or propagating a covered work, you 386 | indicate your acceptance of this License to do so. 387 | 388 | ### 10. Automatic Licensing of Downstream Recipients. 389 | 390 | Each time you convey a covered work, the recipient automatically receives a license 391 | from the original licensors, to run, modify and propagate that work, subject to this 392 | License. You are not responsible for enforcing compliance by third parties with this 393 | License. 394 | 395 | An “entity transaction” is a transaction transferring control of an 396 | organization, or substantially all assets of one, or subdividing an organization, or 397 | merging organizations. If propagation of a covered work results from an entity 398 | transaction, each party to that transaction who receives a copy of the work also 399 | receives whatever licenses to the work the party's predecessor in interest had or 400 | could give under the previous paragraph, plus a right to possession of the 401 | Corresponding Source of the work from the predecessor in interest, if the predecessor 402 | has it or can get it with reasonable efforts. 403 | 404 | You may not impose any further restrictions on the exercise of the rights granted or 405 | affirmed under this License. For example, you may not impose a license fee, royalty, 406 | or other charge for exercise of rights granted under this License, and you may not 407 | initiate litigation (including a cross-claim or counterclaim in a lawsuit) alleging 408 | that any patent claim is infringed by making, using, selling, offering for sale, or 409 | importing the Program or any portion of it. 410 | 411 | ### 11. Patents. 412 | 413 | A “contributor” is a copyright holder who authorizes use under this 414 | License of the Program or a work on which the Program is based. The work thus 415 | licensed is called the contributor's “contributor version”. 416 | 417 | A contributor's “essential patent claims” are all patent claims owned or 418 | controlled by the contributor, whether already acquired or hereafter acquired, that 419 | would be infringed by some manner, permitted by this License, of making, using, or 420 | selling its contributor version, but do not include claims that would be infringed 421 | only as a consequence of further modification of the contributor version. For 422 | purposes of this definition, “control” includes the right to grant patent 423 | sublicenses in a manner consistent with the requirements of this License. 424 | 425 | Each contributor grants you a non-exclusive, worldwide, royalty-free patent license 426 | under the contributor's essential patent claims, to make, use, sell, offer for sale, 427 | import and otherwise run, modify and propagate the contents of its contributor 428 | version. 429 | 430 | In the following three paragraphs, a “patent license” is any express 431 | agreement or commitment, however denominated, not to enforce a patent (such as an 432 | express permission to practice a patent or covenant not to sue for patent 433 | infringement). To “grant” such a patent license to a party means to make 434 | such an agreement or commitment not to enforce a patent against the party. 435 | 436 | If you convey a covered work, knowingly relying on a patent license, and the 437 | Corresponding Source of the work is not available for anyone to copy, free of charge 438 | and under the terms of this License, through a publicly available network server or 439 | other readily accessible means, then you must either (1) cause the Corresponding 440 | Source to be so available, or (2) arrange to deprive yourself of the benefit of the 441 | patent license for this particular work, or (3) arrange, in a manner consistent with 442 | the requirements of this License, to extend the patent license to downstream 443 | recipients. “Knowingly relying” means you have actual knowledge that, but 444 | for the patent license, your conveying the covered work in a country, or your 445 | recipient's use of the covered work in a country, would infringe one or more 446 | identifiable patents in that country that you have reason to believe are valid. 447 | 448 | If, pursuant to or in connection with a single transaction or arrangement, you 449 | convey, or propagate by procuring conveyance of, a covered work, and grant a patent 450 | license to some of the parties receiving the covered work authorizing them to use, 451 | propagate, modify or convey a specific copy of the covered work, then the patent 452 | license you grant is automatically extended to all recipients of the covered work and 453 | works based on it. 454 | 455 | A patent license is “discriminatory” if it does not include within the 456 | scope of its coverage, prohibits the exercise of, or is conditioned on the 457 | non-exercise of one or more of the rights that are specifically granted under this 458 | License. You may not convey a covered work if you are a party to an arrangement with 459 | a third party that is in the business of distributing software, under which you make 460 | payment to the third party based on the extent of your activity of conveying the 461 | work, and under which the third party grants, to any of the parties who would receive 462 | the covered work from you, a discriminatory patent license (a) in connection with 463 | copies of the covered work conveyed by you (or copies made from those copies), or (b) 464 | primarily for and in connection with specific products or compilations that contain 465 | the covered work, unless you entered into that arrangement, or that patent license 466 | was granted, prior to 28 March 2007. 467 | 468 | Nothing in this License shall be construed as excluding or limiting any implied 469 | license or other defenses to infringement that may otherwise be available to you 470 | under applicable patent law. 471 | 472 | ### 12. No Surrender of Others' Freedom. 473 | 474 | If conditions are imposed on you (whether by court order, agreement or otherwise) 475 | that contradict the conditions of this License, they do not excuse you from the 476 | conditions of this License. If you cannot convey a covered work so as to satisfy 477 | simultaneously your obligations under this License and any other pertinent 478 | obligations, then as a consequence you may not convey it at all. For example, if you 479 | agree to terms that obligate you to collect a royalty for further conveying from 480 | those to whom you convey the Program, the only way you could satisfy both those terms 481 | and this License would be to refrain entirely from conveying the Program. 482 | 483 | ### 13. Use with the GNU Affero General Public License. 484 | 485 | Notwithstanding any other provision of this License, you have permission to link or 486 | combine any covered work with a work licensed under version 3 of the GNU Affero 487 | General Public License into a single combined work, and to convey the resulting work. 488 | The terms of this License will continue to apply to the part which is the covered 489 | work, but the special requirements of the GNU Affero General Public License, section 490 | 13, concerning interaction through a network will apply to the combination as such. 491 | 492 | ### 14. Revised Versions of this License. 493 | 494 | The Free Software Foundation may publish revised and/or new versions of the GNU 495 | General Public License from time to time. Such new versions will be similar in spirit 496 | to the present version, but may differ in detail to address new problems or concerns. 497 | 498 | Each version is given a distinguishing version number. If the Program specifies that 499 | a certain numbered version of the GNU General Public License “or any later 500 | version” applies to it, you have the option of following the terms and 501 | conditions either of that numbered version or of any later version published by the 502 | Free Software Foundation. If the Program does not specify a version number of the GNU 503 | General Public License, you may choose any version ever published by the Free 504 | Software Foundation. 505 | 506 | If the Program specifies that a proxy can decide which future versions of the GNU 507 | General Public License can be used, that proxy's public statement of acceptance of a 508 | version permanently authorizes you to choose that version for the Program. 509 | 510 | Later license versions may give you additional or different permissions. However, no 511 | additional obligations are imposed on any author or copyright holder as a result of 512 | your choosing to follow a later version. 513 | 514 | ### 15. Disclaimer of Warranty. 515 | 516 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. 517 | EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES 518 | PROVIDE THE PROGRAM “AS IS” WITHOUT WARRANTY OF ANY KIND, EITHER 519 | EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 520 | MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE 521 | QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE 522 | DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 523 | 524 | ### 16. Limitation of Liability. 525 | 526 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY 527 | COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS THE PROGRAM AS 528 | PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, 529 | INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE 530 | PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE 531 | OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE 532 | WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE 533 | POSSIBILITY OF SUCH DAMAGES. 534 | 535 | ### 17. Interpretation of Sections 15 and 16. 536 | 537 | If the disclaimer of warranty and limitation of liability provided above cannot be 538 | given local legal effect according to their terms, reviewing courts shall apply local 539 | law that most closely approximates an absolute waiver of all civil liability in 540 | connection with the Program, unless a warranty or assumption of liability accompanies 541 | a copy of the Program in return for a fee. 542 | 543 | END OF TERMS AND CONDITIONS 544 | 545 | ## How to Apply These Terms to Your New Programs 546 | 547 | If you develop a new program, and you want it to be of the greatest possible use to 548 | the public, the best way to achieve this is to make it free software which everyone 549 | can redistribute and change under these terms. 550 | 551 | To do so, attach the following notices to the program. It is safest to attach them 552 | to the start of each source file to most effectively state the exclusion of warranty; 553 | and each file should have at least the “copyright” line and a pointer to 554 | where the full notice is found. 555 | 556 | 557 | Copyright (C) 558 | 559 | This program is free software: you can redistribute it and/or modify 560 | it under the terms of the GNU General Public License as published by 561 | the Free Software Foundation, either version 3 of the License, or 562 | (at your option) any later version. 563 | 564 | This program is distributed in the hope that it will be useful, 565 | but WITHOUT ANY WARRANTY; without even the implied warranty of 566 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 567 | GNU General Public License for more details. 568 | 569 | You should have received a copy of the GNU General Public License 570 | along with this program. If not, see . 571 | 572 | Also add information on how to contact you by electronic and paper mail. 573 | 574 | If the program does terminal interaction, make it output a short notice like this 575 | when it starts in an interactive mode: 576 | 577 | Copyright (C) 578 | This program comes with ABSOLUTELY NO WARRANTY; for details type 'show w'. 579 | This is free software, and you are welcome to redistribute it 580 | under certain conditions; type 'show c' for details. 581 | 582 | The hypothetical commands 'show w' and 'show c' should show the appropriate parts of 583 | the General Public License. Of course, your program's commands might be different; 584 | for a GUI interface, you would use an “about box”. 585 | 586 | You should also get your employer (if you work as a programmer) or school, if any, to 587 | sign a “copyright disclaimer” for the program, if necessary. For more 588 | information on this, and how to apply and follow the GNU GPL, see 589 | <>. 590 | 591 | The GNU General Public License does not permit incorporating your program into 592 | proprietary programs. If your program is a subroutine library, you may consider it 593 | more useful to permit linking proprietary applications with the library. If this is 594 | what you want to do, use the GNU Lesser General Public License instead of this 595 | License. But first, please read 596 | <>. --------------------------------------------------------------------------------