mirror of
https://github.com/Iconica-Development/flutter_timeline.git
synced 2025-05-19 10:33:44 +02:00
test
This commit is contained in:
parent
80df20c323
commit
a7b62c4eb5
14 changed files with 573 additions and 0 deletions
43
packages/widgetbook/.gitignore
vendored
Normal file
43
packages/widgetbook/.gitignore
vendored
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
# Miscellaneous
|
||||||
|
*.class
|
||||||
|
*.log
|
||||||
|
*.pyc
|
||||||
|
*.swp
|
||||||
|
.DS_Store
|
||||||
|
.atom/
|
||||||
|
.buildlog/
|
||||||
|
.history
|
||||||
|
.svn/
|
||||||
|
migrate_working_dir/
|
||||||
|
|
||||||
|
# IntelliJ related
|
||||||
|
*.iml
|
||||||
|
*.ipr
|
||||||
|
*.iws
|
||||||
|
.idea/
|
||||||
|
|
||||||
|
# The .vscode folder contains launch configuration and tasks you configure in
|
||||||
|
# VS Code which you may wish to be included in version control, so this line
|
||||||
|
# is commented out by default.
|
||||||
|
#.vscode/
|
||||||
|
|
||||||
|
# Flutter/Dart/Pub related
|
||||||
|
**/doc/api/
|
||||||
|
**/ios/Flutter/.last_build_id
|
||||||
|
.dart_tool/
|
||||||
|
.flutter-plugins
|
||||||
|
.flutter-plugins-dependencies
|
||||||
|
.pub-cache/
|
||||||
|
.pub/
|
||||||
|
/build/
|
||||||
|
|
||||||
|
# Symbolication related
|
||||||
|
app.*.symbols
|
||||||
|
|
||||||
|
# Obfuscation related
|
||||||
|
app.*.map.json
|
||||||
|
|
||||||
|
# Android Studio will place build artifacts here
|
||||||
|
/android/app/debug
|
||||||
|
/android/app/profile
|
||||||
|
/android/app/release
|
16
packages/widgetbook/README.md
Normal file
16
packages/widgetbook/README.md
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
# widgetbook
|
||||||
|
|
||||||
|
A new Flutter project.
|
||||||
|
|
||||||
|
## Getting Started
|
||||||
|
|
||||||
|
This project is a starting point for a Flutter application.
|
||||||
|
|
||||||
|
A few resources to get you started if this is your first Flutter project:
|
||||||
|
|
||||||
|
- [Lab: Write your first Flutter app](https://docs.flutter.dev/get-started/codelab)
|
||||||
|
- [Cookbook: Useful Flutter samples](https://docs.flutter.dev/cookbook)
|
||||||
|
|
||||||
|
For help getting started with Flutter development, view the
|
||||||
|
[online documentation](https://docs.flutter.dev/), which offers tutorials,
|
||||||
|
samples, guidance on mobile development, and a full API reference.
|
28
packages/widgetbook/analysis_options.yaml
Normal file
28
packages/widgetbook/analysis_options.yaml
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
# This file configures the analyzer, which statically analyzes Dart code to
|
||||||
|
# check for errors, warnings, and lints.
|
||||||
|
#
|
||||||
|
# The issues identified by the analyzer are surfaced in the UI of Dart-enabled
|
||||||
|
# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be
|
||||||
|
# invoked from the command line by running `flutter analyze`.
|
||||||
|
|
||||||
|
# The following line activates a set of recommended lints for Flutter apps,
|
||||||
|
# packages, and plugins designed to encourage good coding practices.
|
||||||
|
include: package:flutter_lints/flutter.yaml
|
||||||
|
|
||||||
|
linter:
|
||||||
|
# The lint rules applied to this project can be customized in the
|
||||||
|
# section below to disable rules from the `package:flutter_lints/flutter.yaml`
|
||||||
|
# included above or to enable additional rules. A list of all available lints
|
||||||
|
# and their documentation is published at https://dart.dev/lints.
|
||||||
|
#
|
||||||
|
# Instead of disabling a lint rule for the entire project in the
|
||||||
|
# section below, it can also be suppressed for a single line of code
|
||||||
|
# or a specific dart file by using the `// ignore: name_of_lint` and
|
||||||
|
# `// ignore_for_file: name_of_lint` syntax on the line or in the file
|
||||||
|
# producing the lint.
|
||||||
|
rules:
|
||||||
|
# avoid_print: false # Uncomment to disable the `avoid_print` rule
|
||||||
|
# prefer_single_quotes: true # Uncomment to enable the `prefer_single_quotes` rule
|
||||||
|
|
||||||
|
# Additional information about this file can be found at
|
||||||
|
# https://dart.dev/guides/language/analysis-options
|
91
packages/widgetbook/lib/main.dart
Normal file
91
packages/widgetbook/lib/main.dart
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_timeline/flutter_timeline.dart';
|
||||||
|
import 'package:intl/date_symbol_data_local.dart';
|
||||||
|
import 'package:timeline_widgetbook/main.directories.g.dart';
|
||||||
|
import 'package:timeline_widgetbook/mock_timeline_service.dart';
|
||||||
|
import 'package:widgetbook/widgetbook.dart';
|
||||||
|
import 'package:widgetbook_annotation/widgetbook_annotation.dart' as widgetbook;
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
initializeDateFormatting();
|
||||||
|
|
||||||
|
runApp(const WidgetBookApp());
|
||||||
|
}
|
||||||
|
|
||||||
|
@widgetbook.App()
|
||||||
|
class WidgetBookApp extends StatelessWidget {
|
||||||
|
const WidgetBookApp({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Widgetbook.material(
|
||||||
|
integrations: [
|
||||||
|
WidgetbookCloudIntegration(),
|
||||||
|
],
|
||||||
|
addons: [
|
||||||
|
DeviceFrameAddon(
|
||||||
|
devices: [
|
||||||
|
Devices.ios.iPhoneSE,
|
||||||
|
Devices.ios.iPhone13,
|
||||||
|
Devices.android.bigPhone,
|
||||||
|
Devices.android.mediumPhone,
|
||||||
|
Devices.android.smallPhone,
|
||||||
|
],
|
||||||
|
initialDevice: Devices.ios.iPhone13,
|
||||||
|
),
|
||||||
|
MaterialThemeAddon(
|
||||||
|
themes: [
|
||||||
|
WidgetbookTheme(
|
||||||
|
name: 'Light',
|
||||||
|
data: ThemeData.light(),
|
||||||
|
),
|
||||||
|
WidgetbookTheme(
|
||||||
|
name: 'Dark',
|
||||||
|
data: ThemeData.dark(),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
initialTheme: WidgetbookTheme(
|
||||||
|
name: 'Light',
|
||||||
|
data: ThemeData.light(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
directories: directories,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@widgetbook.UseCase(
|
||||||
|
designLink:
|
||||||
|
'https://www.figma.com/file/PRJoVXQ5aOjAICfkQdAq2A/Iconica-User-Stories?type=design&node-id=34-2763&mode=design&t=W72P3tkEascAKDCk-4',
|
||||||
|
name: 'Timeline post screen',
|
||||||
|
type: TimelinePostScreen,
|
||||||
|
)
|
||||||
|
Widget postScreenUseCase(BuildContext context) {
|
||||||
|
var service = TestTimelineService()..fetchPosts(null);
|
||||||
|
var options = const TimelineOptions();
|
||||||
|
return TimelinePostScreen(
|
||||||
|
userId: '1',
|
||||||
|
service: service,
|
||||||
|
options: options,
|
||||||
|
post: service.posts.first,
|
||||||
|
onPostDelete: () {},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@widgetbook.UseCase(
|
||||||
|
designLink:
|
||||||
|
'https://www.figma.com/file/PRJoVXQ5aOjAICfkQdAq2A/Iconica-User-Stories?type=design&node-id=34-2763&mode=design&t=W72P3tkEascAKDCk-4',
|
||||||
|
name: 'Timeline screen',
|
||||||
|
type: TimelineScreen,
|
||||||
|
)
|
||||||
|
Widget timelineUseCase(BuildContext context) {
|
||||||
|
var service = TestTimelineService()..fetchPosts(null);
|
||||||
|
var options = const TimelineOptions();
|
||||||
|
return TimelineScreen(
|
||||||
|
userId: '1',
|
||||||
|
options: options,
|
||||||
|
onPostTap: (_) {},
|
||||||
|
service: service,
|
||||||
|
);
|
||||||
|
}
|
39
packages/widgetbook/lib/main.directories.g.dart
Normal file
39
packages/widgetbook/lib/main.directories.g.dart
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
// coverage:ignore-file
|
||||||
|
// ignore_for_file: type=lint
|
||||||
|
// ignore_for_file: unused_import, prefer_relative_imports, directives_ordering
|
||||||
|
|
||||||
|
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||||
|
|
||||||
|
// **************************************************************************
|
||||||
|
// AppGenerator
|
||||||
|
// **************************************************************************
|
||||||
|
|
||||||
|
// ignore_for_file: no_leading_underscores_for_library_prefixes
|
||||||
|
import 'package:timeline_widgetbook/main.dart' as _i2;
|
||||||
|
import 'package:widgetbook/widgetbook.dart' as _i1;
|
||||||
|
|
||||||
|
final directories = <_i1.WidgetbookNode>[
|
||||||
|
_i1.WidgetbookFolder(
|
||||||
|
name: 'screens',
|
||||||
|
children: [
|
||||||
|
_i1.WidgetbookLeafComponent(
|
||||||
|
name: 'TimelinePostScreen',
|
||||||
|
useCase: _i1.WidgetbookUseCase(
|
||||||
|
name: 'Timeline post screen',
|
||||||
|
builder: _i2.postScreenUseCase,
|
||||||
|
designLink:
|
||||||
|
'https://www.figma.com/file/PRJoVXQ5aOjAICfkQdAq2A/Iconica-User-Stories?type=design&node-id=34-2763&mode=design&t=W72P3tkEascAKDCk-4',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
_i1.WidgetbookLeafComponent(
|
||||||
|
name: 'TimelineScreen',
|
||||||
|
useCase: _i1.WidgetbookUseCase(
|
||||||
|
name: 'Timeline screen',
|
||||||
|
builder: _i2.timelineUseCase,
|
||||||
|
designLink:
|
||||||
|
'https://www.figma.com/file/PRJoVXQ5aOjAICfkQdAq2A/Iconica-User-Stories?type=design&node-id=34-2763&mode=design&t=W72P3tkEascAKDCk-4',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
];
|
233
packages/widgetbook/lib/mock_timeline_service.dart
Normal file
233
packages/widgetbook/lib/mock_timeline_service.dart
Normal file
|
@ -0,0 +1,233 @@
|
||||||
|
import 'dart:typed_data';
|
||||||
|
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_timeline/flutter_timeline.dart';
|
||||||
|
import 'package:uuid/uuid.dart';
|
||||||
|
|
||||||
|
class TestTimelineService
|
||||||
|
with ChangeNotifier
|
||||||
|
implements TimelineService, TimelineUserService {
|
||||||
|
List<TimelinePost> _posts = [];
|
||||||
|
|
||||||
|
List<TimelinePost> get posts => _posts;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> createPost(TimelinePost post) async {
|
||||||
|
_posts.add(
|
||||||
|
post.copyWith(
|
||||||
|
creator: const TimelinePosterUserModel(userId: 'test_user'),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
notifyListeners();
|
||||||
|
return post;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> deletePost(TimelinePost post) async {
|
||||||
|
_posts = _posts.where((element) => element.id != post.id).toList();
|
||||||
|
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> deletePostReaction(
|
||||||
|
TimelinePost post,
|
||||||
|
String reactionId,
|
||||||
|
) async {
|
||||||
|
if (post.reactions != null && post.reactions!.isNotEmpty) {
|
||||||
|
var reaction =
|
||||||
|
post.reactions!.firstWhere((element) => element.id == reactionId);
|
||||||
|
var updatedPost = post.copyWith(
|
||||||
|
reaction: post.reaction - 1,
|
||||||
|
reactions: (post.reactions ?? [])..remove(reaction),
|
||||||
|
);
|
||||||
|
_posts = _posts
|
||||||
|
.map(
|
||||||
|
(p) => p.id == post.id ? updatedPost : p,
|
||||||
|
)
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
notifyListeners();
|
||||||
|
return updatedPost;
|
||||||
|
}
|
||||||
|
return post;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> fetchPostDetails(TimelinePost post) async {
|
||||||
|
var reactions = post.reactions ?? [];
|
||||||
|
var updatedReactions = <TimelinePostReaction>[];
|
||||||
|
for (var reaction in reactions) {
|
||||||
|
updatedReactions.add(reaction.copyWith(
|
||||||
|
creator: const TimelinePosterUserModel(userId: 'test_user')));
|
||||||
|
}
|
||||||
|
var updatedPost = post.copyWith(reactions: updatedReactions);
|
||||||
|
_posts = _posts.map((p) => (p.id == post.id) ? updatedPost : p).toList();
|
||||||
|
notifyListeners();
|
||||||
|
return updatedPost;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<List<TimelinePost>> fetchPosts(String? category) async {
|
||||||
|
var posts = getMockedPosts();
|
||||||
|
_posts = posts;
|
||||||
|
notifyListeners();
|
||||||
|
return posts;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<List<TimelinePost>> fetchPostsPaginated(
|
||||||
|
String? category,
|
||||||
|
int limit,
|
||||||
|
) async {
|
||||||
|
notifyListeners();
|
||||||
|
return _posts;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> fetchPost(TimelinePost post) async {
|
||||||
|
notifyListeners();
|
||||||
|
return post;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<List<TimelinePost>> refreshPosts(String? category) async {
|
||||||
|
var posts = <TimelinePost>[];
|
||||||
|
|
||||||
|
_posts = [...posts, ..._posts];
|
||||||
|
notifyListeners();
|
||||||
|
return posts;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
TimelinePost? getPost(String postId) =>
|
||||||
|
(_posts.any((element) => element.id == postId))
|
||||||
|
? _posts.firstWhere((element) => element.id == postId)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<TimelinePost> getPosts(String? category) => _posts
|
||||||
|
.where((element) => category == null || element.category == category)
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> likePost(String userId, TimelinePost post) async {
|
||||||
|
var updatedPost = post.copyWith(
|
||||||
|
likes: post.likes + 1,
|
||||||
|
likedBy: (post.likedBy ?? [])..add(userId),
|
||||||
|
);
|
||||||
|
_posts = _posts
|
||||||
|
.map(
|
||||||
|
(p) => p.id == post.id ? updatedPost : p,
|
||||||
|
)
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
notifyListeners();
|
||||||
|
return updatedPost;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> unlikePost(String userId, TimelinePost post) async {
|
||||||
|
var updatedPost = post.copyWith(
|
||||||
|
likes: post.likes - 1,
|
||||||
|
likedBy: post.likedBy?..remove(userId),
|
||||||
|
);
|
||||||
|
_posts = _posts
|
||||||
|
.map(
|
||||||
|
(p) => p.id == post.id ? updatedPost : p,
|
||||||
|
)
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
notifyListeners();
|
||||||
|
return updatedPost;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePost> reactToPost(
|
||||||
|
TimelinePost post,
|
||||||
|
TimelinePostReaction reaction, {
|
||||||
|
Uint8List? image,
|
||||||
|
}) async {
|
||||||
|
var reactionId = const Uuid().v4();
|
||||||
|
var updatedReaction = reaction.copyWith(
|
||||||
|
id: reactionId,
|
||||||
|
creator: const TimelinePosterUserModel(userId: 'test_user'));
|
||||||
|
|
||||||
|
var updatedPost = post.copyWith(
|
||||||
|
reaction: post.reaction + 1,
|
||||||
|
reactions: post.reactions?..add(updatedReaction),
|
||||||
|
);
|
||||||
|
|
||||||
|
_posts = _posts
|
||||||
|
.map(
|
||||||
|
(p) => p.id == post.id ? updatedPost : p,
|
||||||
|
)
|
||||||
|
.toList();
|
||||||
|
notifyListeners();
|
||||||
|
return updatedPost;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<TimelinePost> getMockedPosts() {
|
||||||
|
return [
|
||||||
|
for (var i = 0; i < 20; i++) ...[
|
||||||
|
if (i == 0) ...[
|
||||||
|
TimelinePost(
|
||||||
|
id: 'Post$i',
|
||||||
|
creatorId: 'test_user',
|
||||||
|
title: 'Post $i',
|
||||||
|
category: 'text',
|
||||||
|
content: "Post $i content",
|
||||||
|
likes: i,
|
||||||
|
reaction: 0,
|
||||||
|
reactions: getMockedReactions('Post$i'),
|
||||||
|
createdAt: DateTime.now().subtract(Duration(days: i % 10)),
|
||||||
|
reactionEnabled: true,
|
||||||
|
imageUrl: 'https://picsum.photos/seed/$i/200/300',
|
||||||
|
)
|
||||||
|
] else ...[
|
||||||
|
TimelinePost(
|
||||||
|
id: 'Post$i',
|
||||||
|
creatorId: 'test_user',
|
||||||
|
title: 'Post $i',
|
||||||
|
category: 'text',
|
||||||
|
content: "Post $i content",
|
||||||
|
likes: i,
|
||||||
|
reaction: 0,
|
||||||
|
createdAt: DateTime.now().subtract(Duration(days: i % 10)),
|
||||||
|
reactionEnabled: false,
|
||||||
|
imageUrl: 'https://picsum.photos/seed/$i/200/300',
|
||||||
|
)
|
||||||
|
],
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
List<TimelinePostReaction> getMockedReactions(String posdId) {
|
||||||
|
return [
|
||||||
|
for (var i = 0; i < 20; i++) ...[
|
||||||
|
TimelinePostReaction(
|
||||||
|
id: 'Reaction$i',
|
||||||
|
postId: posdId,
|
||||||
|
reaction: 'Reaction $i',
|
||||||
|
createdAt: DateTime.now().subtract(Duration(days: i % 10)),
|
||||||
|
creatorId: 'test_user',
|
||||||
|
imageUrl:
|
||||||
|
(i % 2 == 0) ? 'https://picsum.photos/seed/$i/200/300' : null,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<TimelinePosterUserModel?> getUser(String userId) async {
|
||||||
|
return TimelinePosterUserModel(
|
||||||
|
userId: userId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
set posts(List<TimelinePost> posts) {
|
||||||
|
_posts = posts;
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
}
|
29
packages/widgetbook/pubspec.yaml
Normal file
29
packages/widgetbook/pubspec.yaml
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
name: timeline_widgetbook
|
||||||
|
description: "A new Flutter project."
|
||||||
|
publish_to: 'none'
|
||||||
|
version: 1.0.0+1
|
||||||
|
|
||||||
|
environment:
|
||||||
|
sdk: '>=3.2.5 <4.0.0'
|
||||||
|
|
||||||
|
dependencies:
|
||||||
|
flutter:
|
||||||
|
sdk: flutter
|
||||||
|
|
||||||
|
cupertino_icons: ^1.0.2
|
||||||
|
widgetbook_annotation: ^3.1.0
|
||||||
|
widgetbook: ^3.7.1
|
||||||
|
flutter_timeline:
|
||||||
|
path: ../flutter_timeline
|
||||||
|
intl: ^0.19.0
|
||||||
|
|
||||||
|
dev_dependencies:
|
||||||
|
build_runner: any
|
||||||
|
widgetbook_generator: ^3.7.0
|
||||||
|
flutter_test:
|
||||||
|
sdk: flutter
|
||||||
|
|
||||||
|
flutter_lints: ^2.0.0
|
||||||
|
|
||||||
|
flutter:
|
||||||
|
uses-material-design: true
|
BIN
packages/widgetbook/web/favicon.png
Normal file
BIN
packages/widgetbook/web/favicon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 917 B |
BIN
packages/widgetbook/web/icons/Icon-192.png
Normal file
BIN
packages/widgetbook/web/icons/Icon-192.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 5.2 KiB |
BIN
packages/widgetbook/web/icons/Icon-512.png
Normal file
BIN
packages/widgetbook/web/icons/Icon-512.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 8.1 KiB |
BIN
packages/widgetbook/web/icons/Icon-maskable-192.png
Normal file
BIN
packages/widgetbook/web/icons/Icon-maskable-192.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 5.5 KiB |
BIN
packages/widgetbook/web/icons/Icon-maskable-512.png
Normal file
BIN
packages/widgetbook/web/icons/Icon-maskable-512.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 20 KiB |
59
packages/widgetbook/web/index.html
Normal file
59
packages/widgetbook/web/index.html
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<!--
|
||||||
|
If you are serving your web app in a path other than the root, change the
|
||||||
|
href value below to reflect the base path you are serving from.
|
||||||
|
|
||||||
|
The path provided below has to start and end with a slash "/" in order for
|
||||||
|
it to work correctly.
|
||||||
|
|
||||||
|
For more details:
|
||||||
|
* https://developer.mozilla.org/en-US/docs/Web/HTML/Element/base
|
||||||
|
|
||||||
|
This is a placeholder for base href that will be replaced by the value of
|
||||||
|
the `--base-href` argument provided to `flutter build`.
|
||||||
|
-->
|
||||||
|
<base href="$FLUTTER_BASE_HREF">
|
||||||
|
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta content="IE=Edge" http-equiv="X-UA-Compatible">
|
||||||
|
<meta name="description" content="A new Flutter project.">
|
||||||
|
|
||||||
|
<!-- iOS meta tags & icons -->
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
||||||
|
<meta name="apple-mobile-web-app-title" content="widgetbook">
|
||||||
|
<link rel="apple-touch-icon" href="icons/Icon-192.png">
|
||||||
|
|
||||||
|
<!-- Favicon -->
|
||||||
|
<link rel="icon" type="image/png" href="favicon.png"/>
|
||||||
|
|
||||||
|
<title>widgetbook</title>
|
||||||
|
<link rel="manifest" href="manifest.json">
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// The value below is injected by flutter build, do not touch.
|
||||||
|
const serviceWorkerVersion = null;
|
||||||
|
</script>
|
||||||
|
<!-- This script adds the flutter initialization JS code -->
|
||||||
|
<script src="flutter.js" defer></script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<script>
|
||||||
|
window.addEventListener('load', function(ev) {
|
||||||
|
// Download main.dart.js
|
||||||
|
_flutter.loader.loadEntrypoint({
|
||||||
|
serviceWorker: {
|
||||||
|
serviceWorkerVersion: serviceWorkerVersion,
|
||||||
|
},
|
||||||
|
onEntrypointLoaded: function(engineInitializer) {
|
||||||
|
engineInitializer.initializeEngine().then(function(appRunner) {
|
||||||
|
appRunner.runApp();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
35
packages/widgetbook/web/manifest.json
Normal file
35
packages/widgetbook/web/manifest.json
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
{
|
||||||
|
"name": "widgetbook",
|
||||||
|
"short_name": "widgetbook",
|
||||||
|
"start_url": ".",
|
||||||
|
"display": "standalone",
|
||||||
|
"background_color": "#0175C2",
|
||||||
|
"theme_color": "#0175C2",
|
||||||
|
"description": "A new Flutter project.",
|
||||||
|
"orientation": "portrait-primary",
|
||||||
|
"prefer_related_applications": false,
|
||||||
|
"icons": [
|
||||||
|
{
|
||||||
|
"src": "icons/Icon-192.png",
|
||||||
|
"sizes": "192x192",
|
||||||
|
"type": "image/png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "icons/Icon-512.png",
|
||||||
|
"sizes": "512x512",
|
||||||
|
"type": "image/png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "icons/Icon-maskable-192.png",
|
||||||
|
"sizes": "192x192",
|
||||||
|
"type": "image/png",
|
||||||
|
"purpose": "maskable"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "icons/Icon-maskable-512.png",
|
||||||
|
"sizes": "512x512",
|
||||||
|
"type": "image/png",
|
||||||
|
"purpose": "maskable"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
Loading…
Reference in a new issue