Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/hogart/sugar-cube-utils

Collection of SugarCube 2 macros and goodies
https://github.com/hogart/sugar-cube-utils

interactive-fiction sugarcube2 twine2

Last synced: 3 months ago
JSON representation

Collection of SugarCube 2 macros and goodies

Awesome Lists containing this project

README

        

# Collection of [SugarCube 2](http://www.motoslave.net/sugarcube/2/) macros and goodies

Please note that code will not work in older browsers (Internet Explorer, pre-Chromium Edge, etc) as-is. See [Using the code](#using-the-code) section below.

If you use [Twee 3 Language Tools](https://github.com/cyrusfirheir/twee3-language-tools) VS Code extension, you might find [`scUtils.twee-config.yaml`](./scUtils.twee-config.yaml) useful, as it contains declarations of all macros in this repo.

---

## Installation

The `*.js` files follow a general pattern of `.js` (and possibly `.css`) for each macro, and the section explaining each macro links to the corresponding file.

To install:
- If using the Twine desktop/web app, copy contents of the `*.js` file to `Story JavaScript`, and if relevant, contents of `*.css` file to `Story Stylesheet`.
- If using a compiler like Tweego, drop the `*.js` — if relevant, `*.css` — files to your source folder.

---

## Contents

### Macros

- [`<>` — `<>`](#anchor-more)
- [`<>` — `<>` — `<>`](#anchor-dlg)
- [`<>` — `<>`](#anchor-gender)
- [`<>`](#anchor-hubnav)
- [`<>`](#anchor-iconcheck)
- [`<>` — `<>` — `<>`](#anchor-journal)
- [`<>`](#anchor-linkif)
- [`<>`](#anchor-linkonce)
- [`<>`](#anchor-rumble)

### Goodies

- [`Achievements`](#anchor-achievments)
- [`menuButton.js`](#anchor-menubutton)
- [`about.js`](#anchor-about)
- [`daymode.js` — `daymode.css`](#anchor-daymode)
- [`faint.js`](#anchor-faint)
- [`fontSize.js`](#anchor-fontsize)
- [`fullscreen.js`](#anchor-fullscreen)
- [`LocationFinder.js`](#anchor-locationfinder)
- [`mute.js`](#anchor-mute)
- [`pickUnique.js`](#anchor-pickunique)
- [`plurals-en.js` — `plurals-ru.js`](#anchor-plurals)
- [`preloadImages.js`](#anchor-preloadimages)
- [`promiseLock.js`](#anchor-promiselock)
- [`qbn.js`](#anchor-qbn)
- [`skipIntro.js`](#anchor-skipintro)
- [`volumeButtons.js`](#anchor-volumebuttons)

---

## Macros

Most macros include built-in styles created by JS, so you can copy-paste one file instead of two. Styles are as unobtrusive and neutral as possible.

---

### `<>` and `<>text<>`

Source: [./more.js](./more.js)

**NB: `<>` is deprecated, use `<>`.**

`<>` shows tooltip — on mouse hover on desktop and on tap on touch screen. It tries to contain the tooltip entirely on the screen. `<>` can include SugarCube code inside it, but not in tooltip content.

---

### `<>`, `<>` and `<>`.

Source: [./dlg.js](./dlg.js)

`<>` is split into `<>`s, which, in turn, consist of `<>`s. It will show the player all lines from given level. After player chooses a line, line's content gets appended and next level is displayed.

```
::Start
— Hi!

<>\
<>\
<>
— Hi, how are you?
— Everything's great! You?
<>
<>
— Hi!
— How are you?
<>
<>\
<>\
<>
— Things are tough, my pet hamster just died!:(
<>
<>
— Things are fine, wanna hang out?
<>
<>
— Sry, playing new game, talk later!
<>
<>
<>
```

#### `<>`
`<>` accepts 3 optional arguments:
* dialogue id (any unique string): needed only if there are several dialogues in the same passage.
* starting level (number): if you need to skip introductions. Defaults to 0.
* prefix (any string): will be prepended to each line. Empty string by default.

Additionally, the macro behavior can be fine-tuned by changing the options passed into the script: `}({trim: false, append: false}));`.
Setting `trim` to `true` will force `<>` to trim its contents before displaying (no unneeded line breaks).
Setting `prepend` to `true` will make `<>` to prepend this line's visible part to the contents before displaying. There will be linebreak between the visible part and the contents.

#### `<>`
`<>` accepts single numeric mandatory argument: it's level.

#### `<>`
`<>` to show after this line, defaults to next level.
`<>` is a container macro; it's content is displayed to player after they choose given line.

Use variables and `<>`s to create branching.

---

### `<>` and `<>`

Source: [./genderswitch.js](./genderswitch.js)

English grammar is pretty neutral when it comes to gender, but other languages are less forgiving. For instance, in Slavic languages you need to put adjectives and past tense verbs in proper grammatical gender.

`<>` displays link-like text which user can click to switch between genders:
`My name is <> Watson.`. Please note that you need to declare `$isFemale` variable in `StoryInit`.

`<>` chooses text between female and male version: `Your father says: My dear <>!`. `<>` assigns `gender-f` and `gender-m` classes to `html` element, so `<>` displays changes reactively, and you can use these classes to further customise game look should you need that.

This can also be used to tell a story from perspectives of two persons regardless of grammatical gender.

---

### `<>`

Source: [./hubnav.js](./hubnav.js)

Easy navigation for set of interconnected locations. Imagine house consisting of bedroom, bathroom, living room, garage and kitchen. Each room has links to other ones but not on itself. Throw in some links that are displayed conditionally and there's whole mess on your hands. `<>` to the rescue!
```
::_home navigation
<>

::Bedroom
Spacious bedroom.
<>

::Bathroom
Take a shower to gain energy. Take a bath to calm down.
<>

::Living room
Your collection of game consoles and huge flatscreen TV.
<>

::Garage
You bought this Vespa trading in retro games.
<>

::Kitchen
Clean, squeky clean, operating room clean.
<>

::Back yard
Daylight keeps vampires at bay. You don't go here at night.
<>
```

Now it's easy to add/delete/rename rooms and change conditions.

#### Tag-based navigation

Even easier option is to pass a string as single argument. `<>` will build navigation based on passages marked with given tag. You can't display link conditionally in this case.

```
::Bedroom [house]
Spacious bedroom.

::Bathroom [house]
Take a shower to gain energy. Take a bath to calm down.

::Living room [house]
Your collection of game consoles and huge flatscreen TV.

::Garage [house]
You bought this Vespa trading in retro games.

::Kitchen [house]
Clean, squeky clean, operating room clean.
```

---

### `<>`

Source: [./iconcheckbox.js](./iconcheckbox.js)

Default checkboxes can look ugly and not fit into overall visual style. So `<>` displays neat switch icon in the same style as built-in SugarCube controls.
The simplest form is `<>toggle value<>`, this will display same label no matter what the value is.
Most flexible form looks like this and allows you to run some callback when value changes:
```
<><>
```

---

### `<>`, `<>` and `<>`

Source: [./journal.js](./journal.js)

Journals/logs/notes/codexes are found in many games.

```
<>Lives on North Pole (this is journal entry content)<>
<>Has 4 reindeers<>
<>Gift giver (this serves as optional title)<>

<>Doesn't exist!!!<>
<><>

<>(Nothing will be shown when journaldisplay called)<>

All arguments are optional and defaults to empty strings
<>Have all journal entries in one place<>
<><>

Entries content gets rendered when <> is used, not when they are added:
<>
<>I like $melike!<>
<>
<>Me<>

```

This can also serve as a simple inventory system.

---

### ``<>``

Source: [./linkif.js](./linkif.js)

Functionally identical to native [`<>`](http://www.motoslave.net/sugarcube/2/docs/#macros-macro-link), except it's only clickable if second argument is truthy, and just shows plain text otherwise.

```
::Room
There's closed <><>. There's also <><>.

::Cupboard
There's key here!
<>
<>
```

Supports any **wiki** form of `[[link|Passage]]`.

---

### `<>`

Source: [./linkonce.js](./linkonce.js)

Functionally identical to ``<>``: shows link if only given passage haven't been visited in current playthrough. Supports any **wiki** form of `[[link|Passage]]`.

---

### `<>`

Source: [./rumble.js](./rumble.js)

Makes your device vibrate, and does nothing if browser/device doesn't [support](https://caniuse.com/#feat=vibration) [Vibration API](https://developer.mozilla.org/docs/Web/API/Navigator/vibrate). Please keep in mind that support is spotty (mostly Android and iOS Chrome and Firefox, no gamepads at all), and long vibrations or sequences can be chopped or dropped entirely, so don't rely on it to convey critical parts of story. Also be polite and provide players with means to turn it off completely.

```
<>
<>
<> or <>
```

Some browsers require user interaction to vibrate, so you'll probably need to wrap this in `<>`.

---

## Goodies

Most goodies/utils put functions into `window.scUtils` "namespace". Things that create buttons in UIBar rely on `menuButton.js`, so include it in your script before.

---

### Achievements

Source: [./achievements](./achievements)

It's not that difficult to create an achievement system, but good achievement system includes many moving parts.
scUtils' `achievements` is capable:

* to provide achievements with title, description, unlock date and flexible checks;
* to have hidden achievements
* to store unlocked achievement between playthroughs (not portable between devices)
* to display achievements as a floating notification in the right bottom corner of player's screen
* to add a button to the sidebar, which displays a dialog containing list of achievements

Take a look inside the [`./achievements/story.twee`](./achievements/story.twee) to learn how to integrate it in your game.

---

### `menuButton.js`

Source: [./menuButton.js](./menuButton.js)

Provide `scUtils.createPassageButton(label, iconContent, passageName)` and `scUtils.createHandlerButton(label, iconContent, shortName, handler)` functions. First one creates button which displays dialogue window displaying some passage content.

---

### `about.js`

Source: [./about.js](./about.js) (relies on menuButton.js)

If story have passage titled `StoryAbout`, adds "About" button which displays dialogue window with this passage rendered inside. Good for providing links to your website/patreon and attributing used assets. To change button label, assign value to correspondent l10n key:`l10nStrings.uiBarAbout = 'Who made this wonderful game?'`

---

### `daymode.js` and `daymode.css`

Source: [./daymode.js](./daymode.js) and [./daymode.css](./daymode.css) (relies on menuButton.js)

Depending on players reading habits, level of fatigue, device, environment and other things author can't predict it can be strainous for eyes to read both white on black or black on white. So let them invert theme when they see fit. Daymode switches your game between default (white on black) theme and (slightly adapted) official but not included bleached.css.

---

### `faint.js`

Source: [./faint.js](./faint.js)

Exposes `scUtils.faint(callback, duration, color, blur)` function, which fills screen with solid `color`, `blur`ring content at the same time and calls `callback` after `duration` seconds. Default values are `faint(callback = null, duration = 5, color = 'black', blur = true)`. Keep in mind that not all browsers support this blurring.

Useful for emulating loosing conscience, teleportation, extended periods of time passing, etc.

---

### `fontSize.js`

Source: [./fontSize.js](./fontSize.js) (relies on menuButton.js)

Exposes `scUtils.createFontSizeBtn` function. When called, this function creates buttons in the sidebar to increase/decrease font size. To change button label, assign value to correspondent l10n key:`l10nStrings.uiFontSize = 'Zoom in/out'` (defaults to 'Font size').

---

### `fullscreen.js`

Source: [./fullscreen.js](./fullscreen.js) (relies on menuButton.js)

Adds "Full screen" button switch to UIBar (if browser supports this API). Supposedly increases immersion. To change button label, assign value to correspondent l10n key:`l10nStrings.uiFullScreen = 'Immersive mode'`.

---

### `LocationFinder.js`

Source: [./LocationFinder.js](./LocationFinder.js)

Game can consist of different locations. Suppose you want to change some styles and switch background music depending on whether player is in a dungeon, forest or desert. Using vanilla SugarCube you'll need to assign designated tag to every passage in each location (and 100 passages is not a very big game). Now add music to equation and remember that player can save/load and use checkpoints.

`scUtils.LocationFinder` tries to solve this issue.
0. Tag passages where player enters new locations with `locationName-desert`, `locationName-forest`, and so on.
0. Call `window.finder = new scUtils.LocationFinder(onChange, 'location-', passageEvents)` in `StoryInit` or in game JavaScript.
0. `onChange` is an optional handler which will be called each time location changes, and is passed `newLocation` and `oldLocation` arguments. You can pass `null` if you don't need this.
0. `'location-'` is an optional prefix to CSS class which will be assigned to `html` according to current location. If you don't need this behavior, pass `null` instead.
0. `passageEvents` is an optional object, mapping [passage events](http://www.motoslave.net/sugarcube/2/docs/passage-events-task-objects.html#passage-events) to handlers (`{':passagestart'(location, event) { console.log(location) }}`). Each handler will receive current location as first argument and original event as second, so you can do some pretty advanced stuff in there.

**NB: Previously LocationFinder needed `locationOrder-` tags and didn't really support open-world games where player could move freely. Now LocationFinder uses history-based "location" detection, so it's not a problem anymore.**

---

### `mute.js`

Source: [./mute.js](./mute.js) (relies on menuButton.js)

**NB: Deprecated, use [volumeButtons.js](#volume-buttons)**

Adds "Sound" button switch to UIBar, which mutes/unmutes SugarCube audio engine (note id doesn't stop playback). To change button label, assign value to correspondent l10n key:`l10nStrings.uiBarMute = 'Shut up'`.

---

### `pickUnique.js`

Source: [./pickUnique.js](./pickUnique.js)

[`.random()`](http://www.motoslave.net/sugarcube/2/docs/#methods-array-prototype-method-random) is a great tool, but sometimes returns same result several times in a row. These helpers solve this problem: each result is guaranteed to be different from previous
```js
// No same food for two days in a row!
const todayIPick = scUtils.pickUnique(['Apple pie', 'Pizza', 'Ice cream', 'Berries']);

// Create a picker function
const foodPicker = scUtils.createUniquePicker(['Apple pie', 'Pizza', 'Ice cream', 'Berries']);
foodPicker(); // function which returns non-repeating results
```

Take a passage, split it into lines and use this lines to produce random non-repeating results. Useful if you have huge list, or the list has long lines.

```
::Script [script]
const picker = scUtils.createUniquePickerFromPassage('Foods')

:: Foods
Apple pie
Pizza
Ice cream
Berries
```

---

### `plurals-en.js` and `plurals-ru.js`

Source: [./plurals.js](./plurals.js) and [./plurals-ru.js](./plurals-ru.js)

While English (and most Germanic and Latin languages) only has two plural forms -- singular, and, well, plural, other languages can have more complex rules. For instance, Slavic languages have 3 forms (for 1 item, for 2..5 items, for lots of items, and they start to repeat when you reach 21), and that's not the limit: Arabic has 6 such forms. So to avoid things like "You have 0 message(s)", you need some utility function. `scUtils.pluralize` and `scUtils.pluralizeFmt` provide that.

`scUtils.pluralize` takes array of cases and amount: `scUtils.pluralize(['cat', 'cats'], numberOfCats)` or `scUtils.pluralize(['яблоко', 'яблок', 'яблока'], numberOfApples)` and return proper string. `scUtils.pluralizeFmt` takes array of cases and template and returns a function which takes number and returns cases and number wrapped in template:
```js
var bulletAmount = window.scUtils.pluralizeFmt(['патрон', 'патрона', 'патронов'], 'У вас ${amount} ${plural}.');
bulletAmount(10); // -> "У вас 10 патронов."
bulletAmount(1); // -> "У вас 1 патрон."
bulletAmount(3); // -> "У вас 3 патрона."
```

Both **plurals-en.js** and **plurals-ru.js** expose same two functions, difference is **plurals.js** works only in pretty recent Chrome versions and support both English and Russian; it requires `lang="en"` attribute on `` to detect language. **plurals-ru.js** works pretty much everywhere but only supports Russian (should work with any Slavic language actually).

**NB: [`plurals.js`](./plurals.js) contains language independent pluralizer, based on built-in [Intl.PluralRules](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Intl/PluralRules). Unfortunately, [browser support is still spotty](https://caniuse.com/intl-pluralrules), so it's not really ready for production.**

---

### `preloadImages.js`

Source: [./preloadImages.js](./preloadImages.js)

If your logo or background image is heavy, it may take some time to load for a player, resulting in unpleasant effect. You may wish to load images beforehand; `scUtils.preloadImages` shows load screen during this process. It also returns a `Promise` for a finer control on what happens next.
```js
scUtils.preloadImages(['img/heavy-background.jsp', 'img/huge-logo.png']).then(() => console.log('Images loaded!'));
```

---

### `promiseLock.js`

Source: [./promiseLock.js](./promiseLock.js)

If you need to load heave scripts, styles, images or other assets, it may be good idea to show load screen during the process. `window.scUtils.promiseLock` accepts a `Promise` object and shows load screen until the promise resolves successfully.

---

### `qbn.js`

Source: [./qbn.js](./qbn.js)

Quest tracker, exposes `window.qbn` and `window.scUtils.qbn` objects.

Suppose player should visit any 5 rooms out of 7 in building before he can proceed with the story, or examine any 3 evidences out of 5 before character comes to conclusion. If character can visit rooms (and return to where they were before) or examine clues in random order, you'll need 7 (or 5) boolean variables and unmaintainable `if()` condition to allow that. You can put some flags into array, but this requires filtering out non-unique values. `qbn` helps with all that:
```js
qbn.set('house', 'ground floor');
qbn.set('house', ['basement', 'kitchen']);
if (qbn.length('house') === 3) { alert('house fully explored') }
qbn.set('house', 'ground floor'); // qbn.length('dungeon') still equals 2

// these methods more useful when tracking some character qualities (like in Sunless Sea)
qbn.unset('island', 'pleasant acquaintance');
qbn.inc('madness', 12); // time to eat your crew yet?
qbn.dec('madness', 5); // qbn.length('madness') === 7
```

---

### `skipIntro.js`

Source: [./skipIntro.js](./skipIntro.js)

Simple utility that inserts "Skip intro" link into certain passages starting from 2nd playthrough.
```js
scUtils.skipIntro(
'My first action passage', // passage to jump to
'Skip boring stuff', // label for the link, defaults to l10nStrings.uiSkipIntro or 'Skip intro'
['Start'], // passages **names** which shouldn't have this link (e.g. splash screen)
['no-skip-intro'], // passages **tags** which shouldn't have this link
)
```
Comes with predefined styles, which can be easily overridden:
```css
p.skipIntro {
text-align: center;
font-size: 300%;
}
```

---

### `volumeButtons.js`

Source: [./volumeButtons.js](./volumeButtons.js) (relies on menuButton.js)

Exposes `scUtils.createVolumeButtons` function, which adds volume control buttons to the UI bar. To change button label, assign value to correspondent l10n key:`l10nStrings.uiVolumeControl = 'Level of AWESOME'`.

```js
const step = 0.2; // volume changes from 0 to 1.
const labels = ['🔈', '🔇', '🔊'];
scUtils.createVolumeButtons(step, volume);
```

---

## Using the code

Code in the repo uses pretty new JS language features and as-is will work only in fresh Chrome and FF and latest Safari (last 2-3 years). This is fine if you're wrapping your game in NW.js or Electron or during debug stages, but may be unacceptable for web distribution. To remedy that, use `bin/build.js` script like so:
```sh
node bin/build.js --es6 abbr about faint genderswitch
```

This will create `./bundle.js` combining transpiled `abbr.js`, `about.js` `faint.js` and `genderswitch.js` files. Additionally, you can produce a minified version adding `--compress` option:
```sh
node bin/build.js --es6 --compress abbr about faint genderswitch
```

By default, code will be transpiled to support same browsers as SugarCube. If you don't have node.js installed, you can transpile code [online](http://babeljs.io/repl/).

---

## MIT License
Copyright 2017-2020 Konstantin Kitmanov.

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.