Skip to content

Commit

Permalink
Smart URLs and Translations
Browse files Browse the repository at this point in the history
  • Loading branch information
Spomky committed Jan 18, 2024
1 parent bd558a6 commit 6c506be
Show file tree
Hide file tree
Showing 21 changed files with 298 additions and 337 deletions.
135 changes: 47 additions & 88 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ Please have a look at the [Web app manifests](https://developer.mozilla.org/en-U
Install the bundle with Composer:

```bash
composer require --dev spomky-labs/phpwa
composer require spomky-labs/phpwa
```

This project follows the [semantic versioning](http://semver.org/) strictly.
Expand Down Expand Up @@ -69,12 +69,22 @@ pwa:
'application/json': ['.json']
```
### Using the Manifest
The manifest can be used in your HTML pages with the following Twig function in the `<head>` section.
It will automatically add the manifest your HTML pages and any other useful meta tags.

```html
{{ pwa() }}
```

### Manifest Generation

When the configuration is set, you can generate the manifest with the following command:
On `dev` or `test` environment, the manifest will be generated for you.
On `prod` environment, the manifest is compiled during the deployment with Asset Mapper.

```bash
symfony console pwa:build
symfony console asset-map:compile
```

By default, the manifest will be generated in the `public` directory with the name `/site.webmanifest`.
Expand All @@ -83,96 +93,49 @@ You can change the output file name and the output folder with the following con
```yaml
# config/packages/phpwa.yaml
pwa:
manifest_filepath: '%kernel.project_dir%/public/sub-folder/pwa.json'
manifest_public_url: '/foo/pwa.json'
```
### Using the Manifest
The manifest can be used in your HTML pages with the following code in the `<head>` section.
In you customized the output filename or the public folder, please replace `/site.webmanifest` with the path to your manifest file.

```html
<link rel="manifest" href="/site.webmanifest">
```

### Manifest Icons

The bundle is able to generate icons from a source image.
This is applicable to the application icons, shortcut icons or Windows 10 widget icons members of the manifest.
The icons must be square and the source image should be at best quality as possible.

To process the icons, you should set an icon processor. The bundle provides a GD processor and an Imagick processor.
Depending on your system, you may have to install one extension or the other.
### Manifest Icons and Screenshots

The bundle is able to link your assets to the manifest file.
Please note that the icons of a size greater than 1024px may be ignored by the browser.

```yaml
# config/packages/phpwa.yaml
pwa:
image_processor: 'pwa.image_processor.gd' # or 'pwa.image_processor.imagick'
icons:
- src: "%kernel.project_dir%/assets/images/logo.png"
sizes: [48, 57, 60, 72, 76, 96, 114, 128, 144, 152, 180, 192, 256, 384, 512, 1024]
format: 'webp'
- src: "%kernel.project_dir%/assets/images/mask.png"
sizes: [48, 57, 60, 72, 76, 96, 114, 128, 144, 152, 180, 192, 256, 384, 512, 1024]
purpose: 'maskable'
- src: "%kernel.project_dir%/assets/images/logo.svg"
- src: "images/logo.png"
sizes: [48, 96, 128, 256, 512, 1024]
- src: "images/logo.svg"
sizes: [0] # 0 means `any` size and is suitable for vector images
- src: "images/logo.svg"
purpose: 'maskable'
sizes: [0]
screenshots:
- src: "screenshots/android_dashboard.png"
platform: 'android'
label: "View of the dashboard on Android"
- "screenshots/android_feature1.png"
- "screenshots/android_feature2.png"
shortcuts:
- name: "Shortcut 1"
short_name: "shortcut-1"
url: "/shortcut1"
description: "Shortcut 1 description"
icons:
- src: "%kernel.project_dir%/assets/images/shortcut1.png"
sizes: [48, 72, 96, 128, 144, 192, 256, 384, 512]
format: 'webp'
- src: "images/shortcut1.png"
sizes: [48, 96, 128, 256, 512, 1024]
- name: "Shortcut 2"
short_name: "shortcut-2"
url: "/shortcut2"
description: "Shortcut 2 description"
icons:
- src: "%kernel.project_dir%/assets/images/shortcut2.png"
sizes: [48, 72, 96, 128, 144, 192, 256, 384, 512]
format: 'webp'
- src: "images/shortcut2.png"
sizes: [48, 96, 128, 256, 512, 1024]
```
With the configuration above, the bundle will generate
* 16 icons from the `logo.png` image. The icons will be converted from `png` to `webp`.
* 16 icons from the `mask.png` image. The format will be `png` and the purpose will be `maskable`.
* And 1 icon from the `logo.svg` image. The format will be `svg` and the size will be `any`.

If the `format` member is present, the bundle will convert the image to the specified format.
In the example above, the `logo.png` image will be converted to `webp`, but the `mask.png` image will not be converted.

### Manifest Screenshots

The bundle is able to generate screenshots from a source image.
This is applicable to the application screenshots or Windows 10 widget screenshot members of the manifest.
Similar to icons, the source image should be at best quality as possible.

You can select a folder where the source images are stored.
The bundle will automatically generate screenshots from all the images in the folder.

```yaml
# config/packages/phpwa.yaml
pwa:
image_processor: 'pwa.image_processor.gd' # or 'pwa.image_processor.imagick'
screenshots:
- src: "%kernel.project_dir%/assets/screenshots/narrow/"
form_factor: "narrow"
- src: "%kernel.project_dir%/assets/screenshots/wide/"
form_factor: "wide"
- src: "%kernel.project_dir%/assets/screenshots/android_dashboard.png"
platform: 'android'
format: 'webp'
label: "View of the dashboard on Android"
```

The bundle will automatically generate screenshots from the source images and add additional information in the manifest
such as the `sizes` and the `form_factor` (`wide` or `narrow`).
The `format` parameter is optional. It indicates the format of the generated image. If not set, the format will be the same as the source image.

### Manifest Shortcuts
The `shortcuts` member may contain a list of action shortcuts that point to specific URLs in your application.
Expand All @@ -194,21 +157,28 @@ pwa:

## Service Worker

The following command will generate a Service Worker in the `public` directory with the name `/sw.js`:
The service worker is a JavaScript file that is executed by the browser.
It can be served by Asset Mapper.

```bash
symfony console pwa:sw
```
```yaml
# config/packages/phpwa.yaml
pwa:
serviceworker: 'script/service-worker.js'
If you want override the existing Service Worker, you can use the `--force` option:
```

```bash
symfony console pwa:sw --force
```yaml
#The following configuration is similar
pwa:
serviceworker:
src: 'script/service-worker.js'
dest: '/sw.js'
scope: '/'
```

Next, you have to register the Service Worker in your HTML pages with the following code in the `<head>` section.
It can also be done in a JavaScript file such as `app.js`.
In you customized the output filename or the public folder, please replace `sw.js` with the path to your Service Worker file.
In you customized the destination filename, please replace `/sw.js` with the path to your Service Worker file.

```html
<script>
Expand All @@ -220,20 +190,9 @@ In you customized the output filename or the public folder, please replace `sw.j
</script>
```

The location of the Service Worker is important. It must be at the root of your application.
In addition, the `serviceworker.src` member of the manifest must be set to the same location.
The `serviceworker.scope` member may be set to the same location or to a sub-folder.
Do not forget to update the `scope` member in the JS configuration.

```yaml
# config/packages/phpwa.yaml
pwa:
serviceworker:
filepath: '%kernel.project_dir%/public/sub-folder/service-worker.js'
src: '/sub-folder/service-worker.js'
scope: '/'
```

### Service Worker Configuration

The Service Worker uses Workbox and comes with predefined configuration and recipes.
Expand Down
1 change: 1 addition & 0 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@
"symfony/mime": "^6.4|^7.0",
"symfony/panther": "^2.1",
"symfony/phpunit-bridge": "^6.4|^7.0",
"symfony/translation": "^7.0",
"symfony/yaml": "^6.4|^7.0",
"symplify/easy-coding-standard": "^12.0"
},
Expand Down
101 changes: 47 additions & 54 deletions src/DependencyInjection/Configuration.php
Original file line number Diff line number Diff line change
Expand Up @@ -63,18 +63,7 @@ private function setupShortcuts(ArrayNodeDefinition $node): void
->info('The description of the shortcut.')
->example('Awesome shortcut')
->end()
->scalarNode('url')
->isRequired()
->info('The URL of the shortcut.')
->example('https://example.com')
->end()
->arrayNode('url_params')
->treatFalseLike([])
->treatTrueLike([])
->treatNullLike([])
->prototype('variable')->end()
->info('The parameters of the action. Only used if the action is a route to a controller.')
->end()
->append($this->getUrlNode('url', 'The URL of the shortcut.'))
->append($this->getIconsNode('The icons of the shortcut.'))
->end()
->end()
Expand Down Expand Up @@ -103,18 +92,7 @@ private function setupFileHandlers(ArrayNodeDefinition $node): void
->treatNullLike([])
->arrayPrototype()
->children()
->scalarNode('action')
->isRequired()
->info('The action to take.')
->example('/handle-audio-file')
->end()
->arrayNode('action_params')
->treatFalseLike([])
->treatTrueLike([])
->treatNullLike([])
->prototype('variable')->end()
->info('The parameters of the action. Only used if the action is a route to a controller.')
->end()
->append($this->getUrlNode('action', 'The action to take.', ['/handle-audio-file']))
->arrayNode('accept')
->requiresAtLeastOneElement()
->useAttributeAsKey('name')
Expand All @@ -140,18 +118,9 @@ private function setupSharedTarget(ArrayNodeDefinition $node): void
->treatNullLike([])
->info('The share target of the application.')
->children()
->scalarNode('action')
->isRequired()
->info('The action of the share target.')
->example('/shared-content-receiver/')
->end()
->arrayNode('action_params')
->treatFalseLike([])
->treatTrueLike([])
->treatNullLike([])
->prototype('variable')->end()
->info('The parameters of the action. Only used if the action is a route to a controller.')
->end()
->append(
$this->getUrlNode('action', 'The action of the share target.', ['/shared-content-receiver/'])
)
->scalarNode('method')
->info('The method of the share target.')
->example('GET')
Expand Down Expand Up @@ -211,18 +180,7 @@ private function setupProtocolHandlers(ArrayNodeDefinition $node): void
->info('The protocol of the handler.')
->example('web+jngl')
->end()
->scalarNode('url')
->isRequired()
->info('The URL of the handler.')
->example('/lookup?type=%s')
->end()
->arrayNode('url_params')
->treatFalseLike([])
->treatTrueLike([])
->treatNullLike([])
->prototype('variable')->end()
->info('The parameters of the action. Only used if the action is a route to a controller.')
->end()
->append($this->getUrlNode('url', 'The URL of the handler.'))
->end()
->end()
->end()
Expand Down Expand Up @@ -269,11 +227,11 @@ private function setupRelatedApplications(ArrayNodeDefinition $node): void
->info('The platform of the application.')
->example('play')
->end()
->scalarNode('url')
->isRequired()
->info('The URL of the application.')
->example('https://play.google.com/store/apps/details?id=com.example.app1')
->end()
->append(
$this->getUrlNode('url', 'The URL of the application.', [
'https://play.google.com/store/apps/details?id=com.example.app1',
])
)
->scalarNode('id')
->info('The ID of the application.')
->example('com.example.app1')
Expand Down Expand Up @@ -314,7 +272,7 @@ private function setupSimpleOptions(ArrayNodeDefinition $node): void
->end()
->end()
->scalarNode('manifest_public_url')
->defaultValue('/site.manifest')
->defaultValue('/site.webmanifest')
->cannotBeEmpty()
->info('The public URL of the manifest file.')
->example('/site.manifest')
Expand Down Expand Up @@ -639,4 +597,39 @@ private function setupWidgets(ArrayNodeDefinition $node): void
->end()
->end();
}

/**
* @param array<string> $examples
*/
private function getUrlNode(string $name, string $info, null|array $examples = null): ArrayNodeDefinition
{
$treeBuilder = new TreeBuilder($name);
$node = $treeBuilder->getRootNode();
assert($node instanceof ArrayNodeDefinition);
$node
->info($info)
->beforeNormalization()
->ifString()
->then(static fn (string $v): array => [
'path' => $v,
])
->end()
->children()
->scalarNode('path')
->isRequired()
->info('The URL of the shortcut.')
->example($examples ?? ['https://example.com', 'app_action_route', '/do/action'])
->end()
->arrayNode('params')
->treatFalseLike([])
->treatTrueLike([])
->treatNullLike([])
->prototype('variable')->end()
->info('The parameters of the action. Only used if the action is a route to a controller.')
->end()
->end()
->end();

return $node;
}
}
10 changes: 1 addition & 9 deletions src/Dto/FileHandler.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,17 +4,9 @@

namespace SpomkyLabs\PwaBundle\Dto;

use Symfony\Component\Serializer\Attribute\SerializedName;

final class FileHandler
{
public string $action;

/**
* @var array<string, mixed>
*/
#[SerializedName('action_params')]
public array $actionParameters = [];
public Url $action;

/**
* @var array<string, string[]>
Expand Down
Loading

0 comments on commit 6c506be

Please sign in to comment.