├── .editorconfig
├── .github
└── ISSUE_TEMPLATE
│ ├── bug_report.md
│ └── feature_request.md
├── .gitignore
├── .npmrc
├── .prettierignore
├── .prettierrc
├── .vscode
└── launch.json
├── CHANGELOG.md
├── CODE_OF_CONDUCT.md
├── CONTRIBUTING.md
├── Jenkinsfile
├── LICENSE
├── README.md
├── android
├── .gitignore
├── app
│ ├── .npmignore
│ ├── build.gradle
│ ├── capacitor.build.gradle
│ ├── proguard-rules.pro
│ ├── release
│ │ └── output.json
│ └── src
│ │ ├── androidTest
│ │ └── java
│ │ │ └── com
│ │ │ └── getcapacitor
│ │ │ └── myapp
│ │ │ └── ExampleInstrumentedTest.java
│ │ ├── main
│ │ ├── AndroidManifest.xml
│ │ ├── assets
│ │ │ └── capacitor.config.json
│ │ ├── ic_launcher-web.png
│ │ ├── java
│ │ │ └── com
│ │ │ │ └── nerdiccoder
│ │ │ │ └── blockphotos
│ │ │ │ └── MainActivity.java
│ │ └── res
│ │ │ ├── drawable-land-hdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-land-ldpi
│ │ │ └── splash.png
│ │ │ ├── drawable-land-mdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-land-xhdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-land-xxhdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-land-xxxhdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-port-hdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-port-ldpi
│ │ │ └── splash.png
│ │ │ ├── drawable-port-mdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-port-xhdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-port-xxhdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-port-xxxhdpi
│ │ │ └── splash.png
│ │ │ ├── drawable-v24
│ │ │ └── ic_launcher_foreground.xml
│ │ │ ├── drawable
│ │ │ ├── ic_launcher_background.xml
│ │ │ └── launch_splash.xml
│ │ │ ├── layout
│ │ │ └── activity_main.xml
│ │ │ ├── mipmap-anydpi-v26
│ │ │ ├── ic_launcher.xml
│ │ │ └── ic_launcher_round.xml
│ │ │ ├── mipmap-hdpi
│ │ │ ├── ic_launcher.png
│ │ │ ├── ic_launcher_foreground.png
│ │ │ ├── ic_launcher_round.png
│ │ │ └── icon.png
│ │ │ ├── mipmap-mdpi
│ │ │ ├── ic_launcher.png
│ │ │ ├── ic_launcher_foreground.png
│ │ │ ├── ic_launcher_round.png
│ │ │ └── icon.png
│ │ │ ├── mipmap-xhdpi
│ │ │ ├── ic_launcher.png
│ │ │ ├── ic_launcher_foreground.png
│ │ │ ├── ic_launcher_round.png
│ │ │ └── icon.png
│ │ │ ├── mipmap-xxhdpi
│ │ │ ├── ic_launcher.png
│ │ │ ├── ic_launcher_foreground.png
│ │ │ ├── ic_launcher_round.png
│ │ │ └── icon.png
│ │ │ ├── mipmap-xxxhdpi
│ │ │ ├── ic_launcher.png
│ │ │ ├── ic_launcher_foreground.png
│ │ │ ├── ic_launcher_round.png
│ │ │ └── icon.png
│ │ │ ├── values
│ │ │ ├── ic_launcher_background.xml
│ │ │ ├── strings.xml
│ │ │ └── styles.xml
│ │ │ └── xml
│ │ │ ├── config.xml
│ │ │ └── file_paths.xml
│ │ └── test
│ │ └── java
│ │ └── com
│ │ └── getcapacitor
│ │ └── myapp
│ │ └── ExampleUnitTest.java
├── build.gradle
├── capacitor.settings.gradle
├── gradle.properties
├── gradle
│ └── wrapper
│ │ ├── gradle-wrapper.jar
│ │ └── gradle-wrapper.properties
├── gradlew
├── gradlew.bat
└── settings.gradle
├── capacitor.config.json
├── electron
├── capacitor.config.json
├── icons
│ ├── mac
│ │ ├── icon.hqx
│ │ ├── icon.icns
│ │ └── icon.ico
│ ├── png
│ │ ├── 1024x1024.png
│ │ ├── 128x128.png
│ │ ├── 16x16.png
│ │ ├── 24x24.png
│ │ ├── 256x256.png
│ │ ├── 32x32.png
│ │ ├── 48x48.png
│ │ ├── 512x512.png
│ │ ├── 64x64.png
│ │ └── 96x96.png
│ └── win
│ │ └── 1024.png.ico
└── main.js
├── ios
├── .gitignore
├── App
│ ├── App.xcodeproj
│ │ ├── project.pbxproj
│ │ └── project.xcworkspace
│ │ │ └── contents.xcworkspacedata
│ ├── App.xcworkspace
│ │ ├── contents.xcworkspacedata
│ │ └── xcshareddata
│ │ │ └── IDEWorkspaceChecks.plist
│ ├── App
│ │ ├── App.entitlements
│ │ ├── AppDelegate.swift
│ │ ├── Assets.xcassets
│ │ │ ├── AppIcon.appiconset
│ │ │ │ ├── AppIcon-20x20@1x.png
│ │ │ │ ├── AppIcon-20x20@2x-1.png
│ │ │ │ ├── AppIcon-20x20@2x.png
│ │ │ │ ├── AppIcon-20x20@3x.png
│ │ │ │ ├── AppIcon-29x29@1x.png
│ │ │ │ ├── AppIcon-29x29@2x-1.png
│ │ │ │ ├── AppIcon-29x29@2x.png
│ │ │ │ ├── AppIcon-29x29@3x.png
│ │ │ │ ├── AppIcon-40x40@1x.png
│ │ │ │ ├── AppIcon-40x40@2x-1.png
│ │ │ │ ├── AppIcon-40x40@2x.png
│ │ │ │ ├── AppIcon-40x40@3x.png
│ │ │ │ ├── AppIcon-512@2x.png
│ │ │ │ ├── AppIcon-60x60@2x.png
│ │ │ │ ├── AppIcon-60x60@3x.png
│ │ │ │ ├── AppIcon-76x76@1x.png
│ │ │ │ ├── AppIcon-76x76@2x.png
│ │ │ │ ├── AppIcon-83.5x83.5@2x.png
│ │ │ │ └── Contents.json
│ │ │ ├── Contents.json
│ │ │ └── Splash.imageset
│ │ │ │ ├── Contents.json
│ │ │ │ ├── splash-2732x2732-1.png
│ │ │ │ ├── splash-2732x2732-2.png
│ │ │ │ └── splash-2732x2732.png
│ │ ├── Base.lproj
│ │ │ ├── LaunchScreen.storyboard
│ │ │ └── Main.storyboard
│ │ ├── Info.plist
│ │ ├── capacitor.config.json
│ │ └── config.xml
│ ├── Podfile
│ └── Podfile.lock
└── capacitor-cordova-ios-plugins
│ ├── CordovaPlugins.podspec
│ ├── CordovaPluginsResources.podspec
│ ├── CordovaPluginsStatic.podspec
│ ├── resources
│ └── .gitkeep
│ └── sources
│ └── .gitkeep
├── launch.json
├── nightwatch.json
├── package-lock.json
├── package.json
├── resources
├── btc-qrcode.png
├── etherium-qrcode.png
├── iota-qrcode.png
└── xrp-qrcode.png
├── src
├── _headers
├── _redirects
├── assets
│ ├── blockstack-19.1.0.min.js
│ ├── blockstack-19.2.1.min.js
│ ├── caman.full.min.js
│ ├── icon
│ │ ├── android-chrome-144x144.png
│ │ ├── android-chrome-192x192.png
│ │ ├── android-chrome-256x256.png
│ │ ├── android-chrome-36x36.png
│ │ ├── android-chrome-384x384.png
│ │ ├── android-chrome-48x48.png
│ │ ├── android-chrome-512x512.png
│ │ ├── android-chrome-72x72.png
│ │ ├── android-chrome-96x96.png
│ │ ├── apple-touch-icon-120x120.png
│ │ ├── apple-touch-icon-152x152.png
│ │ ├── apple-touch-icon-180x180.png
│ │ ├── apple-touch-icon-60x60.png
│ │ ├── apple-touch-icon-76x76.png
│ │ ├── apple-touch-icon.png
│ │ ├── favicon-1024x1024.png
│ │ ├── favicon-16x16.png
│ │ ├── favicon-194x194.png
│ │ ├── favicon-32x32.png
│ │ ├── favicon.ico
│ │ ├── icon.png
│ │ ├── mstile-144x144.png
│ │ ├── mstile-150x150.png
│ │ ├── mstile-310x150.png
│ │ ├── mstile-310x310.png
│ │ ├── mstile-70x70.png
│ │ └── safari-pinned-tab.svg
│ └── placeholder-image.jpg
├── browserconfig.xml
├── components.d.ts
├── components
│ ├── block-img.spec.ts
│ ├── block-img.tsx
│ ├── filter-popover.tsx
│ └── select-album.tsx
├── global
│ ├── app.css
│ └── app.ts
├── index.html
├── interfaces.d.ts
├── manifest.json
├── models
│ ├── photo-metadata.ts
│ └── photo-type.ts
├── pages
│ ├── app-albums
│ │ ├── app-albums.e2e.ts
│ │ ├── app-albums.spec.ts
│ │ └── app-albums.tsx
│ ├── app-photo
│ │ ├── app-photo.css
│ │ ├── app-photo.e2e.ts
│ │ ├── app-photo.spec.ts
│ │ └── app-photo.tsx
│ ├── app-photos
│ │ ├── app-photos.css
│ │ ├── app-photos.e2e.ts
│ │ ├── app-photos.spec.ts
│ │ └── app-photos.tsx
│ ├── app-root
│ │ ├── app-root.css
│ │ ├── app-root.e2e.ts
│ │ ├── app-root.spec.ts
│ │ └── app-root.tsx
│ ├── app-settings
│ │ ├── app-settings.e2e.ts
│ │ ├── app-settings.spec.ts
│ │ └── app-settings.tsx
│ └── app-signin
│ │ ├── app-signin.css
│ │ ├── app-signin.e2e.ts
│ │ ├── app-signin.spec.ts
│ │ └── app-signin.tsx
├── redirect.html
├── services
│ ├── albums-service.ts
│ ├── analytics-service.ts
│ ├── cache-service.spec.ts
│ ├── cache-service.ts
│ ├── photos-service.spec.ts
│ ├── photos-service.ts
│ ├── presenting-service.ts
│ ├── settings-service.ts
│ ├── storage-service.ts
│ └── upload-service.ts
├── site.webmanifest
└── sw.js
├── stencil.config.ts
├── tests
└── Signin.js
├── tsconfig.json
├── tslint.json
└── webpack.config.js
/.editorconfig:
--------------------------------------------------------------------------------
1 | # http://editorconfig.org
2 |
3 | root = true
4 |
5 | [*]
6 | charset = utf-8
7 | indent_style = space
8 | indent_size = 2
9 | end_of_line = lf
10 | insert_final_newline = true
11 | trim_trailing_whitespace = true
12 |
13 | [*.md]
14 | insert_final_newline = false
15 | trim_trailing_whitespace = false
16 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/bug_report.md:
--------------------------------------------------------------------------------
1 | ---
2 | name: Bug report
3 | about: Create a report to help us improve
4 | title: ''
5 | labels: ''
6 | assignees: ''
7 |
8 | ---
9 |
10 | **Describe the bug**
11 | A clear and concise description of what the bug is.
12 |
13 | **To Reproduce**
14 | Steps to reproduce the behavior:
15 | 1. Go to '...'
16 | 2. Click on '....'
17 | 3. Scroll down to '....'
18 | 4. See error
19 |
20 | **Expected behavior**
21 | A clear and concise description of what you expected to happen.
22 |
23 | **Screenshots**
24 | If applicable, add screenshots to help explain your problem.
25 |
26 | **Desktop (please complete the following information):**
27 | - OS: [e.g. iOS]
28 | - Browser [e.g. chrome, safari]
29 | - Version [e.g. 22]
30 |
31 | **Smartphone (please complete the following information):**
32 | - Device: [e.g. iPhone6]
33 | - OS: [e.g. iOS8.1]
34 | - Browser [e.g. stock browser, safari]
35 | - Version [e.g. 22]
36 |
37 | **Additional context**
38 | Add any other context about the problem here.
39 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/feature_request.md:
--------------------------------------------------------------------------------
1 | ---
2 | name: Feature request
3 | about: Suggest an idea for this project
4 | title: ''
5 | labels: ''
6 | assignees: ''
7 |
8 | ---
9 |
10 | **Is your feature request related to a problem? Please describe.**
11 | A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
12 |
13 | **Describe the solution you'd like**
14 | A clear and concise description of what you want to happen.
15 |
16 | **Describe alternatives you've considered**
17 | A clear and concise description of any alternative solutions or features you've considered.
18 |
19 | **Additional context**
20 | Add any other context or screenshots about the feature request here.
21 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/ignore-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 |
6 | # testing
7 | /coverage
8 | /cypress
9 | /www
10 |
11 | # production
12 | /build
13 | /electron/app
14 | /out
15 |
16 | # misc
17 | .DS_Store
18 | .env.local
19 | .env.development.local
20 | .env.test.local
21 | .env.production.local
22 |
23 | npm-debug.log*
24 | yarn-debug.log*
25 | yarn-error.log*
26 |
27 | android/.idea/
28 | reports/
29 |
30 | dist/
31 | !www/favicon.ico
32 | www/
33 |
34 | *~
35 | *.sw[mnpcod]
36 | *.log
37 | *.lock
38 | *.tmp
39 | *.tmp.*
40 | log.txt
41 | *.sublime-project
42 | *.sublime-workspace
43 |
44 | .stencil/
45 | .idea/
46 | .vscode/
47 | .sass-cache/
48 | .versions/
49 | node_modules/
50 | $RECYCLE.BIN/
51 |
52 | .DS_Store
53 | Thumbs.db
54 | UserInterfaceState.xcuserstate
55 | .env
56 |
--------------------------------------------------------------------------------
/.npmrc:
--------------------------------------------------------------------------------
1 | registry=https://registry.npmjs.org/
--------------------------------------------------------------------------------
/.prettierignore:
--------------------------------------------------------------------------------
1 | *.jpg
2 | *.xml
3 | *.svg
4 | *.png
5 | *_headers
6 | *_redirects
7 | *.ico
8 | *blockstack.js
9 | *components.d.ts
10 | *index.html
11 |
--------------------------------------------------------------------------------
/.prettierrc:
--------------------------------------------------------------------------------
1 | {
2 | "singleQuote": true
3 | }
--------------------------------------------------------------------------------
/.vscode/launch.json:
--------------------------------------------------------------------------------
1 | {
2 | "version": "0.2.0",
3 | "configurations": [
4 |
5 | {
6 | "name": "iOS Web",
7 | "type": "chrome",
8 | "request": "attach",
9 | "port": 9222,
10 | "url": "https://79.133.3.164:9876/*",
11 | "webRoot": "${workspaceRoot}/src"
12 | }
13 | ]
14 | }
15 |
--------------------------------------------------------------------------------
/CHANGELOG.md:
--------------------------------------------------------------------------------
1 | # Changelog
2 |
3 | ## v4.2.0 (05/06/2019)
4 |
5 | - [**enhancement**] Updating blockstack.js library to version 19.2.1
6 | - [**bug**] Fix deletion of photos from the photo viewer getting stuck on black page
7 |
8 | ## v4.1.0 (01/06/2019)
9 |
10 | - [**enhancement**] New setting for clearing photos cache
11 | - [**enhancement**] Improve compression from iOS
12 |
13 | ## v4.0.0 (31/05/2019)
14 |
15 | - [**enhancement**] #48 Automatically sync photos between devices
16 | - [**enhancement**] #35 Photos can now be downloaded from Block Photos
17 | - [**enhancement**] #50 Compressed versions of photos for faster loading times
18 | - [**enhancement**] Redesign of menu buttons and loaders
19 |
20 | ## v3.0.0 (09/04/2019)
21 |
22 | - [**enhancement**] #26 Photo viewer Slide between pictures with multi-touch
23 | - [**enhancement**] #25 Slide between photos in the Photo viewer with the keyboard arrow keys
24 | - [**enhancement**] #38 Zoom in on photos in the photo viewer
25 | - [**enhancement**] Updating blockstack.js library to version 19.1.0
26 |
27 | ## v2.0.0 (10/01/2019)
28 |
29 | - [**enhancement**] Possibility to create photo albums [#41](https://github.com/nerdic-coder/block-photos/issues/41)
30 | - [**enhancement**] Add upload photos progress bar [#44](https://github.com/nerdic-coder/block-photos/issues/44)
31 | - [**enhancement**] Settings page to turn on/off anonymous tracking data
32 | - [**enhancement**] New side-menu layout
33 |
34 | ## v1.3.0 (10/01/2019)
35 |
36 | - [**bug**] Fix security issue with blockstack by updating blockstack.js library to version 18.2.1.
37 |
38 | ## v1.2.0 (04/12/2018)
39 |
40 | - [**enhancement**] Default text when pictures list is empty [#45](https://github.com/nerdic-coder/block-photos/issues/45)
41 | - [**bug**] Improve the Android PWA icons [#37](https://github.com/nerdic-coder/block-photos/issues/37)
42 | - [**bug**] CSS transform: rotate not rendering correctly on iOS [#34](https://github.com/nerdic-coder/block-photos/issues/34)
43 | - [**enhancement**] Add more details to profile page [#32](https://github.com/nerdic-coder/block-photos/issues/32)
44 | - [**enhancement**] Add the possibility to delete several pictures at once [#31](https://github.com/nerdic-coder/block-photos/issues/31)
45 | - [**enhancement**] Add similar styling to the app as the landing page [#30](https://github.com/nerdic-coder/block-photos/issues/30)
46 | - [**enhancement**] Upload photos in web app version [#29](https://github.com/nerdic-coder/block-photos/issues/29)
47 | - [**closed**] Fix 'Duplicate Code' issue in Profile and Picture pages with uploadFiles method [#27](https://github.com/nerdic-coder/block-photos/issues/27)
48 | - [**closed**] Right click menu in pictures list [#24](https://github.com/nerdic-coder/block-photos/issues/24)
49 | - [**closed**] Icon and logo for the Block Photos app [#23](https://github.com/nerdic-coder/block-photos/issues/23)
50 | - [**closed**] Create unit tests for pages [#22](https://github.com/nerdic-coder/block-photos/issues/22)
51 | - [**closed**] Create unit tests for PictureService class [#21](https://github.com/nerdic-coder/block-photos/issues/21)
52 | - [**closed**] Create blockstack mock for unit tests [#20](https://github.com/nerdic-coder/block-photos/issues/20)
53 | - [**closed**] Create electron mock for unit tests [#19](https://github.com/nerdic-coder/block-photos/issues/19)
54 | - [**closed**] Write unit tests for component BlockImg [#18](https://github.com/nerdic-coder/block-photos/issues/18)
55 | - [**closed**] Add ESLint checks on project code [#17](https://github.com/nerdic-coder/block-photos/issues/17)
56 | - [**closed**] Empty cache on logout [#16](https://github.com/nerdic-coder/block-photos/issues/16)
57 | - [**enhancement**] Add support for running Block Photos on iOS and Android [#15](https://github.com/nerdic-coder/block-photos/issues/15)
58 | - [**closed**] Slide between pictures in the picture view [#13](https://github.com/nerdic-coder/block-photos/issues/13)
59 | - [**closed**] Drag and drop uploads of pictures [#11](https://github.com/nerdic-coder/block-photos/issues/11)
60 | - [**closed**] Refresh button on the pictures list [#10](https://github.com/nerdic-coder/block-photos/issues/10)
61 | - [**enhancement**] Have the pictures list in an endless scroll [#9](https://github.com/nerdic-coder/block-photos/issues/9)
62 | - [**closed**] Show loading spinner on various actions [#8](https://github.com/nerdic-coder/block-photos/issues/8)
63 | - [**enhancement**] Add instructions on login page [#7](https://github.com/nerdic-coder/block-photos/issues/7)
64 | - [**enhancement**] Add a button to rotate the images [#6](https://github.com/nerdic-coder/block-photos/issues/6)
65 | - [**closed**] Error handling at login [#5](https://github.com/nerdic-coder/block-photos/issues/5)
66 | - [**closed**] Error handling when the picutres list can't be loaded [#4](https://github.com/nerdic-coder/block-photos/issues/4)
67 | - [**closed**] Error handling when image uploading goes wrong [#3](https://github.com/nerdic-coder/block-photos/issues/3)
68 | - [**closed**] Add a file size error message [#2](https://github.com/nerdic-coder/block-photos/issues/2)
69 | - [**closed**] Switch cache to Web SQL storage [#1](https://github.com/nerdic-coder/block-photos/issues/1)
70 |
--------------------------------------------------------------------------------
/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
6 | contributors and maintainers pledge to making participation in our project and
7 | our community a harassment-free experience for everyone, regardless of age, body
8 | size, disability, ethnicity, sex characteristics, gender identity and expression,
9 | level of experience, education, socio-economic status, nationality, personal
10 | appearance, race, religion, or sexual identity and orientation.
11 |
12 | ## Our Standards
13 |
14 | Examples of behavior that contributes to creating a positive environment
15 | include:
16 |
17 | * Using welcoming and inclusive language
18 | * Being respectful of differing viewpoints and experiences
19 | * Gracefully accepting constructive criticism
20 | * Focusing on what is best for the community
21 | * Showing empathy towards other community members
22 |
23 | Examples of unacceptable behavior by participants include:
24 |
25 | * The use of sexualized language or imagery and unwelcome sexual attention or
26 | advances
27 | * Trolling, insulting/derogatory comments, and personal or political attacks
28 | * Public or private harassment
29 | * Publishing others' private information, such as a physical or electronic
30 | address, without explicit permission
31 | * Other conduct which could reasonably be considered inappropriate in a
32 | professional setting
33 |
34 | ## Our Responsibilities
35 |
36 | Project maintainers are responsible for clarifying the standards of acceptable
37 | behavior and are expected to take appropriate and fair corrective action in
38 | response to any instances of unacceptable behavior.
39 |
40 | Project maintainers have the right and responsibility to remove, edit, or
41 | reject comments, commits, code, wiki edits, issues, and other contributions
42 | that are not aligned to this Code of Conduct, or to ban temporarily or
43 | permanently any contributor for other behaviors that they deem inappropriate,
44 | threatening, offensive, or harmful.
45 |
46 | ## Scope
47 |
48 | This Code of Conduct applies both within project spaces and in public spaces
49 | when an individual is representing the project or its community. Examples of
50 | representing a project or community include using an official project e-mail
51 | address, posting via an official social media account, or acting as an appointed
52 | representative at an online or offline event. Representation of a project may be
53 | further defined and clarified by project maintainers.
54 |
55 | ## Enforcement
56 |
57 | Instances of abusive, harassing, or otherwise unacceptable behavior may be
58 | reported by contacting the project team at johan@nerdic-coder.com. All
59 | complaints will be reviewed and investigated and will result in a response that
60 | is deemed necessary and appropriate to the circumstances. The project team is
61 | obligated to maintain confidentiality with regard to the reporter of an incident.
62 | Further details of specific enforcement policies may be posted separately.
63 |
64 | Project maintainers who do not follow or enforce the Code of Conduct in good
65 | faith may face temporary or permanent repercussions as determined by other
66 | members of the project's leadership.
67 |
68 | ## Attribution
69 |
70 | This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
71 | available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html
72 |
73 | [homepage]: https://www.contributor-covenant.org
74 |
75 | For answers to common questions about this code of conduct, see
76 | https://www.contributor-covenant.org/faq
77 |
--------------------------------------------------------------------------------
/CONTRIBUTING.md:
--------------------------------------------------------------------------------
1 | ## Contribution to Block Photos
2 |
3 | To contribute you can either report issues like bugs or good new features.
4 | If you are a developer that think you can solve any of our issues, fork this repo, follow the instructions below to develop and test your changes.
5 | Then create a pull request to this repository develop branch. Good luck!
6 |
7 | Installation:
8 |
9 | ```bash
10 | git clone https://github.com/nerdic-coder/block-photos.git block-photos
11 | cd block-photos
12 | npm install
13 | ```
14 |
15 | Starting Test server:
16 |
17 | ```bash
18 | npm run serve
19 | ```
20 |
21 | Starting Electron app:
22 |
23 | ```bash
24 | npm start
25 | ```
26 |
27 | Build and package Electron app:
28 |
29 | ```bash
30 | npm run package
31 | ```
32 |
33 | Build and package for web app distribution:
34 |
35 | ```bash
36 | npm run build:web
37 | ```
38 |
39 | Build for Android app distribution:
40 |
41 | ```bash
42 | npm run build:android
43 | ```
44 |
45 | Build for iOS app distribution:
46 |
47 | ```bash
48 | npm run build:ios
49 | ```
50 |
51 | Test Lint:
52 |
53 | ```bash
54 | npm run lint
55 | ```
56 |
57 | Run unit tests:
58 |
59 | ```bash
60 | npm test
61 | ```
62 |
63 | Run e2e tests:
64 |
65 | 1. Start the apps dev server:
66 |
67 | ```bash
68 | npm run serve
69 | ```
70 |
71 | 2. Start selenium
72 |
73 | ```bash
74 | npm run selenium
75 | ```
76 |
77 | 3. Run the tests
78 |
79 | ```bash
80 | npm run e2e
81 | ```
82 |
--------------------------------------------------------------------------------
/Jenkinsfile:
--------------------------------------------------------------------------------
1 | pipeline {
2 | agent any
3 | environment {
4 | CI = 'true'
5 | }
6 | stages {
7 | stage('Install') {
8 | steps {
9 | bat 'npm install'
10 | }
11 | }
12 | stage('Lint') {
13 | steps {
14 | bat 'npm run lint'
15 | }
16 | }
17 | stage('Test') {
18 | steps {
19 | bat 'npm run test'
20 | }
21 | }
22 | stage('Build') {
23 | steps {
24 | bat 'npm run build'
25 | }
26 | }
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2018 Johan Axelsson
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Block Photos - Your photo album on the blockchain!
2 |
3 |
4 |
5 |
6 |
7 | [](https://github.com/nerdic-coder/block-photos/releases/)
8 | [](https://github.com/nerdic-coder/block-photos/releases/)
9 | [](https://github.com/nerdic-coder/block-photos)
10 | [](https://github.com/nerdic-coder/block-photos)
11 | [](https://github.com/nerdic-coder/block-photos)
12 | [](https://www.codefactor.io/repository/github/nerdic-coder/block-photos/overview/master)
13 | [](https://app.block-photos.com/)
14 | [](https://twitter.com/Block_Photos)
15 |
16 | ## Project links
17 |
18 | Visit the website here: [block-photos.com](https://block-photos.com/)
19 |
20 | Test the web app here: [app.block-photos.com](https://app.block-photos.com/)
21 |
22 | Follow Block Photos on Twitter: [@Block_Photos](https://twitter.com/Block_Photos)
23 |
24 | Read my introduction article about this project: [Building a decentralized Photos app on Blockstack with React](https://nerdic-coder.com/2018/06/22/building-a-decentralized-photos-app-on-blockstack-with-react/)
25 |
26 | ## Donations
27 |
28 | If you like this project please donate to preferred cryptocurrency below:
29 |
30 | ### Coinbase commerce
31 |
32 | [Click here to donate with Coinbase commerce](https://commerce.coinbase.com/checkout/9d35f08b-bd51-40b0-a502-b88250cffc6b)
33 |
34 | ### Bitcoin
35 |
36 | 3JkxUQ763fbf1cYVAitqpWUMeY4QLaBU8M
37 |
38 |
39 |
40 |
41 |
42 | ### Etherium
43 |
44 | 0x6e54183E7b22dA87fb48e0d79ADaDa4f665A4D5a
45 |
46 |
47 |
48 |
49 |
50 | ### XRP
51 |
52 | rGaGGSbeF8HmmrLnTg8gaT92Wa77qg3Y8k
53 |
54 |
55 |
56 |
57 |
58 | ### IOTA
59 |
60 | HTPB9JVTMCDHEDMCIPOJELELVLNWLGMHVGZJUCKOYNBZGN9EVCDKFEIUJKINPCBWZI9BKAVIHRVRNWPFC9CQXJYQSZ
61 |
62 |
63 |
64 |
65 |
66 | ### NEO
67 |
68 | ALNbr6NSuEkWE7ASNB9iB5VcpZmhz1FaAg
69 |
70 | ## Developing the project
71 |
72 | To test this project do the following...
73 |
74 | Installation:
75 |
76 | ```bash
77 | git clone https://github.com/nerdic-coder/block-photos.git block-photos
78 | cd block-photos
79 | npm install
80 | ```
81 |
82 | Starting Test server:
83 |
84 | ```bash
85 | npm run serve
86 | ```
87 |
88 | Starting Electron app:
89 |
90 | ```bash
91 | npm start
92 | ```
93 |
94 | Build and package Electron app:
95 |
96 | ```bash
97 | npm run package
98 | ```
99 |
100 | Build and package for web app distribution:
101 |
102 | ```bash
103 | npm run build:web
104 | ```
105 |
106 | Build for Android app distribution:
107 |
108 | ```bash
109 | npm run build:android
110 | ```
111 |
112 | Build for iOS app distribution:
113 |
114 | ```bash
115 | npm run build:ios
116 | ```
117 |
118 | Test ESLint:
119 |
120 | ```bash
121 | npm run lint
122 | ```
123 |
124 | Run unit tests:
125 |
126 | ```bash
127 | npm test
128 | ```
129 |
130 | Run e2e tests:
131 |
132 | 1. Start the apps dev server:
133 |
134 | ```bash
135 | npm run serve
136 | ```
137 |
138 | 2. Start selenium
139 |
140 | ```bash
141 | npm run selenium
142 | ```
143 |
144 | 3. Run the tests
145 |
146 | ```bash
147 | npm run e2e
148 | ```
149 |
150 | ## Uglify
151 |
152 | Uglify javascript files with: https://www.npmjs.com/package/uglify-es
153 |
--------------------------------------------------------------------------------
/android/.gitignore:
--------------------------------------------------------------------------------
1 | # NPM renames .gitignore to .npmignore
2 | # In order to prevent that, we remove the initial "."
3 | # And the CLI then renames it
4 |
5 | # Using Android gitignore template: https://github.com/github/gitignore/blob/master/Android.gitignore
6 |
7 | # Built application files
8 | *.apk
9 | *.ap_
10 |
11 | # Files for the ART/Dalvik VM
12 | *.dex
13 |
14 | # Java class files
15 | *.class
16 |
17 | # Generated files
18 | bin/
19 | gen/
20 | out/
21 |
22 | # Gradle files
23 | .gradle/
24 | build/
25 |
26 | # Local configuration file (sdk path, etc)
27 | local.properties
28 |
29 | # Proguard folder generated by Eclipse
30 | proguard/
31 |
32 | # Log Files
33 | *.log
34 |
35 | # Android Studio Navigation editor temp files
36 | .navigation/
37 |
38 | # Android Studio captures folder
39 | captures/
40 |
41 | # IntelliJ
42 | *.iml
43 | .idea/workspace.xml
44 | .idea/tasks.xml
45 | .idea/gradle.xml
46 | .idea/dictionaries
47 | .idea/libraries
48 |
49 | # Keystore files
50 | # Uncomment the following line if you do not want to check your keystore files in.
51 | #*.jks
52 |
53 | # External native build folder generated in Android Studio 2.2 and later
54 | .externalNativeBuild
55 |
56 | # Google Services (e.g. APIs or Firebase)
57 | google-services.json
58 |
59 | # Freeline
60 | freeline.py
61 | freeline/
62 | freeline_project_description.json
63 |
64 | # fastlane
65 | fastlane/report.xml
66 | fastlane/Preview.html
67 | fastlane/screenshots
68 | fastlane/test_output
69 | fastlane/readme.md
70 |
71 | # Cordova plugins for Capacitor
72 | capacitor-cordova-android-plugins
73 |
74 | # Copied web assets
75 | app/src/main/assets/public
76 |
--------------------------------------------------------------------------------
/android/app/.npmignore:
--------------------------------------------------------------------------------
1 | /build/*
2 | !/build/.npmkeep
3 |
--------------------------------------------------------------------------------
/android/app/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion 28
5 | defaultConfig {
6 | applicationId "com.nerdiccoder.blockphotos"
7 | minSdkVersion 21
8 | targetSdkVersion 28
9 | versionCode 9
10 | versionName "4.2"
11 | testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
12 | }
13 | buildTypes {
14 | release {
15 | minifyEnabled false
16 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
17 | }
18 | }
19 | lintOptions {
20 | // Or, if you prefer, you can continue to check for errors in release builds,
21 | // but continue the build even when errors are found:
22 | abortOnError false
23 | }
24 | }
25 |
26 | repositories {
27 | maven {
28 | url "https://dl.bintray.com/ionic-team/capacitor"
29 | }
30 | flatDir{
31 | dirs '../capacitor-cordova-android-plugins/src/main/libs', 'libs'
32 | }
33 | }
34 |
35 | dependencies {
36 | implementation fileTree(include: ['*.jar'], dir: 'libs')
37 | implementation 'com.android.support:appcompat-v7:28.0.0'
38 | implementation project(':capacitor-android')
39 | testImplementation 'junit:junit:4.12'
40 | androidTestImplementation 'com.android.support.test:runner:1.0.2'
41 | androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
42 | }
43 |
44 | apply from: 'capacitor.build.gradle'
45 |
46 | try {
47 | def servicesJSON = file('google-services.json')
48 | if (servicesJSON.text) {
49 | apply plugin: 'com.google.gms.google-services'
50 | }
51 | } catch(Exception e) {
52 | logger.warn("google-services.json not found, google-services plugin not applied. Push Notifications won't work")
53 | }
54 |
--------------------------------------------------------------------------------
/android/app/capacitor.build.gradle:
--------------------------------------------------------------------------------
1 | // DO NOT EDIT THIS FILE! IT IS GENERATED EACH TIME "capacitor update" IS RUN
2 |
3 | dependencies {
4 |
5 |
6 | }
7 |
8 |
9 | if (hasProperty('postBuildExtras')) {
10 | postBuildExtras()
11 | }
12 |
--------------------------------------------------------------------------------
/android/app/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # You can control the set of applied configuration files using the
3 | # proguardFiles setting in build.gradle.
4 | #
5 | # For more details, see
6 | # http://developer.android.com/guide/developing/tools/proguard.html
7 |
8 | # If your project uses WebView with JS, uncomment the following
9 | # and specify the fully qualified class name to the JavaScript interface
10 | # class:
11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
12 | # public *;
13 | #}
14 |
15 | # Uncomment this to preserve the line number information for
16 | # debugging stack traces.
17 | #-keepattributes SourceFile,LineNumberTable
18 |
19 | # If you keep the line number information, uncomment this to
20 | # hide the original source file name.
21 | #-renamesourcefileattribute SourceFile
22 |
--------------------------------------------------------------------------------
/android/app/release/output.json:
--------------------------------------------------------------------------------
1 | [
2 | {
3 | "outputType": { "type": "APK" },
4 | "apkInfo": {
5 | "type": "MAIN",
6 | "splits": [],
7 | "versionCode": 8,
8 | "versionName": "4.1",
9 | "enabled": true,
10 | "outputFile": "app-release.apk",
11 | "fullName": "release",
12 | "baseName": "release"
13 | },
14 | "path": "app-release.apk",
15 | "properties": {}
16 | }
17 | ]
18 |
--------------------------------------------------------------------------------
/android/app/src/androidTest/java/com/getcapacitor/myapp/ExampleInstrumentedTest.java:
--------------------------------------------------------------------------------
1 | package com.getcapacitor.myapp;
2 |
3 | import android.content.Context;
4 | import android.support.test.InstrumentationRegistry;
5 | import android.support.test.runner.AndroidJUnit4;
6 |
7 | import org.junit.Test;
8 | import org.junit.runner.RunWith;
9 |
10 | import static org.junit.Assert.*;
11 |
12 | /**
13 | * Instrumented test, which will execute on an Android device.
14 | *
15 | * @see Testing documentation
16 | */
17 | @RunWith(AndroidJUnit4.class)
18 | public class ExampleInstrumentedTest {
19 | @Test
20 | public void useAppContext() throws Exception {
21 | // Context of the app under test.
22 | Context appContext = InstrumentationRegistry.getTargetContext();
23 |
24 | assertEquals("com.getcapacitor.app", appContext.getPackageName());
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/android/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
13 |
14 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
40 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
--------------------------------------------------------------------------------
/android/app/src/main/assets/capacitor.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "appId": "com.nerdiccoder.blockphotos",
3 | "appName": "Block Photos",
4 | "bundledWebRuntime": false,
5 | "webDir": "www"
6 | }
7 |
--------------------------------------------------------------------------------
/android/app/src/main/ic_launcher-web.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/ic_launcher-web.png
--------------------------------------------------------------------------------
/android/app/src/main/java/com/nerdiccoder/blockphotos/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.nerdiccoder.blockphotos;
2 |
3 | import android.os.Bundle;
4 |
5 | import com.getcapacitor.BridgeActivity;
6 | import com.getcapacitor.Plugin;
7 |
8 | import java.util.ArrayList;
9 |
10 | public class MainActivity extends BridgeActivity {
11 | @Override
12 | public void onCreate(Bundle savedInstanceState) {
13 | super.onCreate(savedInstanceState);
14 |
15 | // Initializes the Bridge
16 | this.init(savedInstanceState, new ArrayList>() {{
17 | // Additional plugins you've installed go here
18 | // Ex: add(TotallyAwesomePlugin.class);
19 | }});
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-land-hdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-land-hdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-land-ldpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-land-ldpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-land-mdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-land-mdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-land-xhdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-land-xhdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-land-xxhdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-land-xxhdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-land-xxxhdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-land-xxxhdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-port-hdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-port-hdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-port-ldpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-port-ldpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-port-mdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-port-mdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-port-xhdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-port-xhdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-port-xxhdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-port-xxhdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-port-xxxhdpi/splash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/drawable-port-xxxhdpi/splash.png
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable-v24/ic_launcher_foreground.xml:
--------------------------------------------------------------------------------
1 |
7 |
12 |
13 |
19 |
22 |
25 |
26 |
27 |
28 |
34 |
35 |
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable/ic_launcher_background.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
10 |
15 |
20 |
25 |
30 |
35 |
40 |
45 |
50 |
55 |
60 |
65 |
70 |
75 |
80 |
85 |
90 |
95 |
100 |
105 |
110 |
115 |
120 |
125 |
130 |
135 |
140 |
145 |
150 |
155 |
160 |
165 |
170 |
171 |
--------------------------------------------------------------------------------
/android/app/src/main/res/drawable/launch_splash.xml:
--------------------------------------------------------------------------------
1 |
2 |
--------------------------------------------------------------------------------
/android/app/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
12 |
13 |
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-hdpi/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-hdpi/icon.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-mdpi/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-mdpi/icon.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xhdpi/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xhdpi/icon.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxhdpi/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxhdpi/icon.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/android/app/src/main/res/mipmap-xxxhdpi/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/app/src/main/res/mipmap-xxxhdpi/icon.png
--------------------------------------------------------------------------------
/android/app/src/main/res/values/ic_launcher_background.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #FFFFFF
4 |
--------------------------------------------------------------------------------
/android/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | Block Photos
4 | Block Photos
5 | com.nerdiccoder.blockphotos
6 | com.nerdiccoder.blockphotos.fileprovider
7 | blockphotosapp
8 |
9 |
--------------------------------------------------------------------------------
/android/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
12 |
13 |
18 |
19 |
20 |
23 |
24 |
--------------------------------------------------------------------------------
/android/app/src/main/res/xml/config.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/android/app/src/main/res/xml/file_paths.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/android/app/src/test/java/com/getcapacitor/myapp/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.getcapacitor.myapp;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * Example local unit test, which will execute on the development machine (host).
9 | *
10 | * @see Testing documentation
11 | */
12 | public class ExampleUnitTest {
13 | @Test
14 | public void addition_isCorrect() throws Exception {
15 | assertEquals(4, 2 + 2);
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/android/build.gradle:
--------------------------------------------------------------------------------
1 | // Top-level build file where you can add configuration options common to all sub-projects/modules.
2 |
3 | buildscript {
4 |
5 | repositories {
6 | google()
7 | jcenter()
8 | }
9 | dependencies {
10 | classpath 'com.android.tools.build:gradle:3.3.2'
11 | classpath 'com.google.gms:google-services:4.2.0'
12 |
13 | // NOTE: Do not place your application dependencies here; they belong
14 | // in the individual module build.gradle files
15 | }
16 | }
17 |
18 | allprojects {
19 | repositories {
20 | google()
21 | jcenter()
22 | }
23 | }
24 |
25 | task clean(type: Delete) {
26 | delete rootProject.buildDir
27 | }
28 |
--------------------------------------------------------------------------------
/android/capacitor.settings.gradle:
--------------------------------------------------------------------------------
1 | // DO NOT EDIT THIS FILE! IT IS GENERATED EACH TIME "capacitor update" IS RUN
2 | include ':capacitor-android'
3 | project(':capacitor-android').projectDir = new File('../node_modules/@capacitor/android/capacitor')
4 |
--------------------------------------------------------------------------------
/android/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | org.gradle.jvmargs=-Xmx1536m
13 |
14 | # When configured, Gradle will run in incubating parallel mode.
15 | # This option should only be used with decoupled projects. More details, visit
16 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
17 | # org.gradle.parallel=true
18 |
--------------------------------------------------------------------------------
/android/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/android/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/android/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Tue Jan 30 13:14:22 CST 2018
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-4.10.1-all.zip
7 |
--------------------------------------------------------------------------------
/android/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # Attempt to set APP_HOME
46 | # Resolve links: $0 may be a link
47 | PRG="$0"
48 | # Need this for relative symlinks.
49 | while [ -h "$PRG" ] ; do
50 | ls=`ls -ld "$PRG"`
51 | link=`expr "$ls" : '.*-> \(.*\)$'`
52 | if expr "$link" : '/.*' > /dev/null; then
53 | PRG="$link"
54 | else
55 | PRG=`dirname "$PRG"`"/$link"
56 | fi
57 | done
58 | SAVED="`pwd`"
59 | cd "`dirname \"$PRG\"`/" >/dev/null
60 | APP_HOME="`pwd -P`"
61 | cd "$SAVED" >/dev/null
62 |
63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
64 |
65 | # Determine the Java command to use to start the JVM.
66 | if [ -n "$JAVA_HOME" ] ; then
67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
68 | # IBM's JDK on AIX uses strange locations for the executables
69 | JAVACMD="$JAVA_HOME/jre/sh/java"
70 | else
71 | JAVACMD="$JAVA_HOME/bin/java"
72 | fi
73 | if [ ! -x "$JAVACMD" ] ; then
74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
75 |
76 | Please set the JAVA_HOME variable in your environment to match the
77 | location of your Java installation."
78 | fi
79 | else
80 | JAVACMD="java"
81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
82 |
83 | Please set the JAVA_HOME variable in your environment to match the
84 | location of your Java installation."
85 | fi
86 |
87 | # Increase the maximum file descriptors if we can.
88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
89 | MAX_FD_LIMIT=`ulimit -H -n`
90 | if [ $? -eq 0 ] ; then
91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
92 | MAX_FD="$MAX_FD_LIMIT"
93 | fi
94 | ulimit -n $MAX_FD
95 | if [ $? -ne 0 ] ; then
96 | warn "Could not set maximum file descriptor limit: $MAX_FD"
97 | fi
98 | else
99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
100 | fi
101 | fi
102 |
103 | # For Darwin, add options to specify how the application appears in the dock
104 | if $darwin; then
105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
106 | fi
107 |
108 | # For Cygwin, switch paths to Windows format before running java
109 | if $cygwin ; then
110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
112 | JAVACMD=`cygpath --unix "$JAVACMD"`
113 |
114 | # We build the pattern for arguments to be converted via cygpath
115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
116 | SEP=""
117 | for dir in $ROOTDIRSRAW ; do
118 | ROOTDIRS="$ROOTDIRS$SEP$dir"
119 | SEP="|"
120 | done
121 | OURCYGPATTERN="(^($ROOTDIRS))"
122 | # Add a user-defined pattern to the cygpath arguments
123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
125 | fi
126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
127 | i=0
128 | for arg in "$@" ; do
129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
131 |
132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
134 | else
135 | eval `echo args$i`="\"$arg\""
136 | fi
137 | i=$((i+1))
138 | done
139 | case $i in
140 | (0) set -- ;;
141 | (1) set -- "$args0" ;;
142 | (2) set -- "$args0" "$args1" ;;
143 | (3) set -- "$args0" "$args1" "$args2" ;;
144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
150 | esac
151 | fi
152 |
153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
154 | function splitJvmOpts() {
155 | JVM_OPTS=("$@")
156 | }
157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
159 |
160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
161 |
--------------------------------------------------------------------------------
/android/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/android/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':app'
2 | include ':capacitor-cordova-android-plugins'
3 | project(':capacitor-cordova-android-plugins').projectDir = new File('./capacitor-cordova-android-plugins/')
4 |
5 | apply from: 'capacitor.settings.gradle'
6 |
--------------------------------------------------------------------------------
/capacitor.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "appId": "com.nerdiccoder.blockphotos",
3 | "appName": "Block Photos",
4 | "bundledWebRuntime": false,
5 | "webDir": "www"
6 | }
7 |
--------------------------------------------------------------------------------
/electron/capacitor.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "appId": "com.nerdiccoder.blockphotos",
3 | "appName": "Block Photos",
4 | "bundledWebRuntime": false,
5 | "webDir": "www"
6 | }
7 |
--------------------------------------------------------------------------------
/electron/icons/mac/icon.icns:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/mac/icon.icns
--------------------------------------------------------------------------------
/electron/icons/mac/icon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/mac/icon.ico
--------------------------------------------------------------------------------
/electron/icons/png/1024x1024.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/1024x1024.png
--------------------------------------------------------------------------------
/electron/icons/png/128x128.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/128x128.png
--------------------------------------------------------------------------------
/electron/icons/png/16x16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/16x16.png
--------------------------------------------------------------------------------
/electron/icons/png/24x24.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/24x24.png
--------------------------------------------------------------------------------
/electron/icons/png/256x256.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/256x256.png
--------------------------------------------------------------------------------
/electron/icons/png/32x32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/32x32.png
--------------------------------------------------------------------------------
/electron/icons/png/48x48.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/48x48.png
--------------------------------------------------------------------------------
/electron/icons/png/512x512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/512x512.png
--------------------------------------------------------------------------------
/electron/icons/png/64x64.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/64x64.png
--------------------------------------------------------------------------------
/electron/icons/png/96x96.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/png/96x96.png
--------------------------------------------------------------------------------
/electron/icons/win/1024.png.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/electron/icons/win/1024.png.ico
--------------------------------------------------------------------------------
/electron/main.js:
--------------------------------------------------------------------------------
1 | // Modules to control application life and create native browser window
2 | const { app, BrowserWindow, protocol, dialog, shell } = require('electron');
3 | const path = require('path');
4 | const isDev = require('electron-is-dev');
5 |
6 | let BASE_URL = `file://${__dirname}/app/index.html`;
7 | let SCHEME = 'blockphotosapp';
8 | if (isDev) {
9 | // BASE_URL = 'http://localhost:9876';
10 | }
11 |
12 | protocol.registerStandardSchemes([SCHEME]);
13 | app.setAsDefaultProtocolClient(SCHEME);
14 |
15 | // Keep a global reference of the window object, if you don't, the window will
16 | // be closed automatically when the JavaScript object is garbage collected.
17 | let mainWindow;
18 | let currentAuthResponse = '';
19 |
20 | const gotTheLock = app.requestSingleInstanceLock();
21 |
22 | if (!gotTheLock) {
23 | app.quit();
24 | return;
25 | } else {
26 | app.on('second-instance', (event, commandLine) => {
27 | if (mainWindow && commandLine[1]) {
28 | if (mainWindow.isMinimized()) {
29 | mainWindow.restore();
30 | }
31 |
32 | var request = commandLine[1].split(':');
33 |
34 | if (
35 | request[1] &&
36 | currentAuthResponse !== request[1] &&
37 | request[1].includes('localhost') === false
38 | ) {
39 | currentAuthResponse = request[1];
40 |
41 | secondWindow = new BrowserWindow({
42 | width: 500,
43 | height: 810,
44 | icon: path.join(__dirname, 'icons/png/64x64.png')
45 | });
46 |
47 | secondWindow.focus();
48 | secondWindow.loadURL(BASE_URL + '?authResponse=' + request[1]);
49 | mainWindow.close();
50 |
51 | mainWindow = secondWindow;
52 |
53 | return true;
54 | } else {
55 | shell.openExternal(commandLine[1]);
56 | return true;
57 | }
58 | }
59 | shell.openExternal(commandLine[1]);
60 | // dialog.showMessageBox({
61 | // message: 'Authentication failed, please try again!' + commandLine[1],
62 | // buttons: ['OK']
63 | // });
64 | });
65 | }
66 |
67 | // Handle creating/removing shortcuts on Windows when installing/uninstalling.
68 | if (require('electron-squirrel-startup')) {
69 | app.quit();
70 | }
71 |
72 | const createWindow = () => {
73 | // Create the browser window.
74 | mainWindow = new BrowserWindow({
75 | width: 500,
76 | height: 810,
77 | icon: path.join(__dirname, 'icons/png/64x64.png')
78 | });
79 |
80 | // and load the index.html of the app.
81 | mainWindow.loadURL(BASE_URL);
82 |
83 | // Open the DevTools.
84 | // mainWindow.webContents.openDevTools();
85 |
86 | // Emitted when the window is closed.
87 | mainWindow.on('closed', () => {
88 | // Dereference the window object, usually you would store windows
89 | // in an array if your app supports multi windows, this is the time
90 | // when you should delete the corresponding element.
91 | mainWindow = null;
92 | });
93 | };
94 |
95 | // This method will be called when Electron has finished
96 | // initialization and is ready to create browser windows.
97 | // Some APIs can only be used after this event occurs.
98 | app.on('ready', createWindow);
99 |
100 | // Quit when all windows are closed.
101 | app.on('window-all-closed', function() {
102 | // On OS X it is common for applications and their menu bar
103 | // to stay active until the user quits explicitly with Cmd + Q
104 | if (process.platform !== 'darwin') {
105 | app.quit();
106 | }
107 | });
108 |
109 | app.on('activate', function() {
110 | // On OS X it's common to re-create a window in the app when the
111 | // dock icon is clicked and there are no other windows open.
112 | if (mainWindow === null) {
113 | createWindow();
114 | }
115 | });
116 |
117 | app.on('open-url', function(event, url) {
118 | // On OS X it's common to re-create a window in the app when the
119 | // dock icon is clicked and there are no other windows open.
120 | if (mainWindow) {
121 | if (mainWindow.isMinimized()) {
122 | mainWindow.restore();
123 | }
124 |
125 | var request = url.split(':');
126 |
127 | if (request[1] && currentAuthResponse !== request[1]) {
128 | currentAuthResponse = request[1];
129 |
130 | mainWindow.focus();
131 | mainWindow.loadURL(BASE_URL + '?authResponse=' + request[1]);
132 | return;
133 | }
134 | }
135 | dialog.showMessageBox({
136 | message: 'Authentication failed, please try again!',
137 | buttons: ['OK']
138 | });
139 | });
140 |
141 | // In this file you can include the rest of your app's specific main process
142 | // code. You can also put them in separate files and require them here.
143 |
--------------------------------------------------------------------------------
/ios/.gitignore:
--------------------------------------------------------------------------------
1 | # NPM renames .gitignore to .npmignore
2 | # In order to prevent that, we remove the initial "."
3 | # And the CLI then renames it
4 |
5 | App/build
6 | App/Pods
7 | App/public
8 | xcuserdata
9 |
10 |
--------------------------------------------------------------------------------
/ios/App/App.xcodeproj/project.xcworkspace/contents.xcworkspacedata:
--------------------------------------------------------------------------------
1 |
2 |
4 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/ios/App/App.xcworkspace/contents.xcworkspacedata:
--------------------------------------------------------------------------------
1 |
2 |
4 |
6 |
7 |
9 |
10 |
11 |
--------------------------------------------------------------------------------
/ios/App/App.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | IDEDidComputeMac32BitWarning
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/ios/App/App/App.entitlements:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | aps-environment
6 | development
7 |
8 |
9 |
--------------------------------------------------------------------------------
/ios/App/App/AppDelegate.swift:
--------------------------------------------------------------------------------
1 | import UIKit
2 | import Capacitor
3 |
4 | @UIApplicationMain
5 | class AppDelegate: UIResponder, UIApplicationDelegate {
6 |
7 | var window: UIWindow?
8 |
9 |
10 | func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
11 | // Override point for customization after application launch.
12 | return true
13 | }
14 |
15 | func applicationWillResignActive(_ application: UIApplication) {
16 | // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state.
17 | // Use this method to pause ongoing tasks, disable timers, and invalidate graphics rendering callbacks. Games should use this method to pause the game.
18 | }
19 |
20 | func applicationDidEnterBackground(_ application: UIApplication) {
21 | // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later.
22 | // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits.
23 | }
24 |
25 | func applicationWillEnterForeground(_ application: UIApplication) {
26 | // Called as part of the transition from the background to the active state; here you can undo many of the changes made on entering the background.
27 | }
28 |
29 | func applicationDidBecomeActive(_ application: UIApplication) {
30 | // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
31 | }
32 |
33 | func applicationWillTerminate(_ application: UIApplication) {
34 | // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.
35 | }
36 |
37 | func application(_ app: UIApplication, open url: URL, options: [UIApplicationOpenURLOptionsKey : Any] = [:]) -> Bool {
38 | // Called when the app was launched with a url. Feel free to add additional processing here,
39 | // but if you want the App API to support tracking app url opens, make sure to keep this call
40 | return CAPBridge.handleOpenUrl(url, options)
41 | }
42 |
43 | func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([Any]?) -> Void) -> Bool {
44 | // Called when the app was launched with an activity, including Universal Links.
45 | // Feel free to add additional processing here, but if you want the App API to support
46 | // tracking app url opens, make sure to keep this call
47 | return CAPBridge.handleContinueActivity(userActivity, restorationHandler)
48 | }
49 |
50 | override func touchesBegan(_ touches: Set, with event: UIEvent?) {
51 | super.touchesBegan(touches, with: event)
52 |
53 | let statusBarRect = UIApplication.shared.statusBarFrame
54 | guard let touchPoint = event?.allTouches?.first?.location(in: self.window) else { return }
55 |
56 | if statusBarRect.contains(touchPoint) {
57 | NotificationCenter.default.post(CAPBridge.statusBarTappedNotification)
58 | }
59 | }
60 |
61 | func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) {
62 | NotificationCenter.default.post(name: Notification.Name(CAPNotifications.DidRegisterForRemoteNotificationsWithDeviceToken.name()), object: deviceToken)
63 | }
64 |
65 | func application(_ application: UIApplication, didFailToRegisterForRemoteNotificationsWithError error: Error) {
66 | NotificationCenter.default.post(name: Notification.Name(CAPNotifications.DidFailToRegisterForRemoteNotificationsWithError.name()), object: error)
67 | }
68 |
69 | }
70 |
71 |
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@1x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@1x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@2x-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@2x-1.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@3x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-20x20@3x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@1x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@1x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@2x-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@2x-1.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@3x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-29x29@3x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@1x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@1x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@2x-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@2x-1.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@3x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-40x40@3x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-512@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-512@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-60x60@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-60x60@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-60x60@3x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-60x60@3x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-76x76@1x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-76x76@1x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-76x76@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-76x76@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-83.5x83.5@2x.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-83.5x83.5@2x.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/AppIcon.appiconset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images" : [
3 | {
4 | "size" : "20x20",
5 | "idiom" : "iphone",
6 | "filename" : "AppIcon-20x20@2x.png",
7 | "scale" : "2x"
8 | },
9 | {
10 | "size" : "20x20",
11 | "idiom" : "iphone",
12 | "filename" : "AppIcon-20x20@3x.png",
13 | "scale" : "3x"
14 | },
15 | {
16 | "size" : "29x29",
17 | "idiom" : "iphone",
18 | "filename" : "AppIcon-29x29@2x-1.png",
19 | "scale" : "2x"
20 | },
21 | {
22 | "size" : "29x29",
23 | "idiom" : "iphone",
24 | "filename" : "AppIcon-29x29@3x.png",
25 | "scale" : "3x"
26 | },
27 | {
28 | "size" : "40x40",
29 | "idiom" : "iphone",
30 | "filename" : "AppIcon-40x40@2x.png",
31 | "scale" : "2x"
32 | },
33 | {
34 | "size" : "40x40",
35 | "idiom" : "iphone",
36 | "filename" : "AppIcon-40x40@3x.png",
37 | "scale" : "3x"
38 | },
39 | {
40 | "size" : "60x60",
41 | "idiom" : "iphone",
42 | "filename" : "AppIcon-60x60@2x.png",
43 | "scale" : "2x"
44 | },
45 | {
46 | "size" : "60x60",
47 | "idiom" : "iphone",
48 | "filename" : "AppIcon-60x60@3x.png",
49 | "scale" : "3x"
50 | },
51 | {
52 | "size" : "20x20",
53 | "idiom" : "ipad",
54 | "filename" : "AppIcon-20x20@1x.png",
55 | "scale" : "1x"
56 | },
57 | {
58 | "size" : "20x20",
59 | "idiom" : "ipad",
60 | "filename" : "AppIcon-20x20@2x-1.png",
61 | "scale" : "2x"
62 | },
63 | {
64 | "size" : "29x29",
65 | "idiom" : "ipad",
66 | "filename" : "AppIcon-29x29@1x.png",
67 | "scale" : "1x"
68 | },
69 | {
70 | "size" : "29x29",
71 | "idiom" : "ipad",
72 | "filename" : "AppIcon-29x29@2x.png",
73 | "scale" : "2x"
74 | },
75 | {
76 | "size" : "40x40",
77 | "idiom" : "ipad",
78 | "filename" : "AppIcon-40x40@1x.png",
79 | "scale" : "1x"
80 | },
81 | {
82 | "size" : "40x40",
83 | "idiom" : "ipad",
84 | "filename" : "AppIcon-40x40@2x-1.png",
85 | "scale" : "2x"
86 | },
87 | {
88 | "size" : "76x76",
89 | "idiom" : "ipad",
90 | "filename" : "AppIcon-76x76@1x.png",
91 | "scale" : "1x"
92 | },
93 | {
94 | "size" : "76x76",
95 | "idiom" : "ipad",
96 | "filename" : "AppIcon-76x76@2x.png",
97 | "scale" : "2x"
98 | },
99 | {
100 | "size" : "83.5x83.5",
101 | "idiom" : "ipad",
102 | "filename" : "AppIcon-83.5x83.5@2x.png",
103 | "scale" : "2x"
104 | },
105 | {
106 | "size" : "1024x1024",
107 | "idiom" : "ios-marketing",
108 | "filename" : "AppIcon-512@2x.png",
109 | "scale" : "1x"
110 | }
111 | ],
112 | "info" : {
113 | "version" : 1,
114 | "author" : "xcode"
115 | }
116 | }
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "info" : {
3 | "version" : 1,
4 | "author" : "xcode"
5 | }
6 | }
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/Splash.imageset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images" : [
3 | {
4 | "idiom" : "universal",
5 | "filename" : "splash-2732x2732-2.png",
6 | "scale" : "1x"
7 | },
8 | {
9 | "idiom" : "universal",
10 | "filename" : "splash-2732x2732-1.png",
11 | "scale" : "2x"
12 | },
13 | {
14 | "idiom" : "universal",
15 | "filename" : "splash-2732x2732.png",
16 | "scale" : "3x"
17 | }
18 | ],
19 | "info" : {
20 | "version" : 1,
21 | "author" : "xcode"
22 | }
23 | }
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732-1.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732-2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732-2.png
--------------------------------------------------------------------------------
/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732.png
--------------------------------------------------------------------------------
/ios/App/App/Base.lproj/LaunchScreen.storyboard:
--------------------------------------------------------------------------------
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 |
29 |
30 |
--------------------------------------------------------------------------------
/ios/App/App/Base.lproj/Main.storyboard:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
--------------------------------------------------------------------------------
/ios/App/App/Info.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | CFBundleDevelopmentRegion
6 | en
7 | CFBundleDisplayName
8 | Block Photos
9 | CFBundleExecutable
10 | $(EXECUTABLE_NAME)
11 | CFBundleIdentifier
12 | $(PRODUCT_BUNDLE_IDENTIFIER)
13 | CFBundleInfoDictionaryVersion
14 | 6.0
15 | CFBundleName
16 | $(PRODUCT_NAME)
17 | CFBundlePackageType
18 | APPL
19 | CFBundleShortVersionString
20 | 1.2
21 | CFBundleURLTypes
22 |
23 |
24 | CFBundleTypeRole
25 | Editor
26 | CFBundleURLSchemes
27 |
28 | prefs
29 |
30 |
31 |
32 | CFBundleURLSchemes
33 |
34 | blockphotosapp
35 |
36 |
37 |
38 | CFBundleVersion
39 | 3
40 | LSRequiresIPhoneOS
41 |
42 | NSAppTransportSecurity
43 |
44 | NSAllowsArbitraryLoads
45 |
46 |
47 | NSCameraUsageDescription
48 | To Take Photos and Video
49 | NSLocationAlwaysUsageDescription
50 | Always allow Geolocation?
51 | NSLocationWhenInUseUsageDescription
52 | Allow Geolocation?
53 | NSMicrophoneUsageDescription
54 | To Record Audio With Video
55 | NSPhotoLibraryAddUsageDescription
56 | Store camera photos to camera
57 | NSPhotoLibraryUsageDescription
58 | To Pick Photos from Library
59 | UILaunchStoryboardName
60 | LaunchScreen
61 | UIMainStoryboardFile
62 | Main
63 | UIRequiredDeviceCapabilities
64 |
65 | armv7
66 |
67 | UIStatusBarStyle
68 | UIStatusBarStyleLightContent
69 | UISupportedInterfaceOrientations
70 |
71 | UIInterfaceOrientationPortrait
72 | UIInterfaceOrientationLandscapeLeft
73 | UIInterfaceOrientationLandscapeRight
74 |
75 | UISupportedInterfaceOrientations~ipad
76 |
77 | UIInterfaceOrientationPortrait
78 | UIInterfaceOrientationPortraitUpsideDown
79 | UIInterfaceOrientationLandscapeLeft
80 | UIInterfaceOrientationLandscapeRight
81 |
82 | UIViewControllerBasedStatusBarAppearance
83 |
84 |
85 |
86 |
--------------------------------------------------------------------------------
/ios/App/App/capacitor.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "appId": "com.nerdiccoder.blockphotos",
3 | "appName": "Block Photos",
4 | "bundledWebRuntime": false,
5 | "webDir": "www"
6 | }
7 |
--------------------------------------------------------------------------------
/ios/App/App/config.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/ios/App/Podfile:
--------------------------------------------------------------------------------
1 | platform :ios, '11.0'
2 | use_frameworks!
3 |
4 | target 'App' do
5 | # Add your Pods here
6 |
7 | # Automatic Capacitor Pod dependencies, do not delete
8 | pod 'Capacitor', :path => '../../node_modules/@capacitor/ios'
9 | pod 'CapacitorCordova', :path => '../../node_modules/@capacitor/ios'
10 |
11 | # Do not delete
12 | end
13 |
--------------------------------------------------------------------------------
/ios/App/Podfile.lock:
--------------------------------------------------------------------------------
1 | PODS:
2 | - Capacitor (1.0.0-beta.10):
3 | - CapacitorCordova (= 1.0.0-beta.10)
4 | - CapacitorCordova (1.0.0-beta.10)
5 |
6 | DEPENDENCIES:
7 | - "Capacitor (from `../../node_modules/@capacitor/ios`)"
8 | - "CapacitorCordova (from `../../node_modules/@capacitor/ios`)"
9 |
10 | EXTERNAL SOURCES:
11 | Capacitor:
12 | :path: "../../node_modules/@capacitor/ios"
13 | CapacitorCordova:
14 | :path: "../../node_modules/@capacitor/ios"
15 |
16 | SPEC CHECKSUMS:
17 | Capacitor: 9aa60986d00166b45b3216daf90f49f8df6593fa
18 | CapacitorCordova: aab520b872cabe0d9f30a03dc385c5dce8d2c82b
19 |
20 | PODFILE CHECKSUM: 144b255a8aae841ea14f3ee0bfb55be44afd1167
21 |
22 | COCOAPODS: 1.5.3
23 |
--------------------------------------------------------------------------------
/ios/capacitor-cordova-ios-plugins/CordovaPlugins.podspec:
--------------------------------------------------------------------------------
1 |
2 | Pod::Spec.new do |s|
3 | s.name = 'CordovaPlugins'
4 | s.version = '1.0.0-beta.11'
5 | s.summary = 'Autogenerated spec'
6 | s.license = 'Unknown'
7 | s.homepage = 'https://example.com'
8 | s.authors = { 'Capacitor Generator' => 'hi@example.com' }
9 | s.source = { :git => 'https://github.com/ionic-team/does-not-exist.git', :tag => '1.0.0-beta.11' }
10 | s.source_files = 'sources/**/*.{swift,h,m,c,cc,mm,cpp}'
11 | s.ios.deployment_target = '11.0'
12 | s.dependency 'CapacitorCordova'
13 | s.swift_version = '4.0'
14 |
15 | end
--------------------------------------------------------------------------------
/ios/capacitor-cordova-ios-plugins/CordovaPluginsResources.podspec:
--------------------------------------------------------------------------------
1 | Pod::Spec.new do |s|
2 | s.name = 'CordovaPluginsResources'
3 | s.version = '0.0.105'
4 | s.summary = 'Resources for Cordova plugins'
5 | s.social_media_url = 'http://twitter.com/getcapacitor'
6 | s.license = 'MIT'
7 | s.homepage = 'https://capacitor.ionicframework.com/'
8 | s.authors = { 'Ionic Team' => 'hi@ionicframework.com' }
9 | s.source = { :git => 'https://github.com/ionic-team/capacitor.git', :tag => s.version.to_s }
10 | s.resources = ['resources/*']
11 | end
--------------------------------------------------------------------------------
/ios/capacitor-cordova-ios-plugins/CordovaPluginsStatic.podspec:
--------------------------------------------------------------------------------
1 |
2 | Pod::Spec.new do |s|
3 | s.name = 'CordovaPluginsStatic'
4 | s.version = '1.0.0-beta.11'
5 | s.summary = 'Autogenerated spec'
6 | s.license = 'Unknown'
7 | s.homepage = 'https://example.com'
8 | s.authors = { 'Capacitor Generator' => 'hi@example.com' }
9 | s.source = { :git => 'https://github.com/ionic-team/does-not-exist.git', :tag => '1.0.0-beta.11' }
10 | s.source_files = 'sourcesstatic/**/*.{swift,h,m,c,cc,mm,cpp}'
11 | s.ios.deployment_target = '11.0'
12 | s.dependency 'CapacitorCordova'
13 | s.swift_version = '4.0'
14 | s.static_framework = true
15 | end
--------------------------------------------------------------------------------
/ios/capacitor-cordova-ios-plugins/resources/.gitkeep:
--------------------------------------------------------------------------------
1 |
2 |
--------------------------------------------------------------------------------
/ios/capacitor-cordova-ios-plugins/sources/.gitkeep:
--------------------------------------------------------------------------------
1 |
2 |
--------------------------------------------------------------------------------
/launch.json:
--------------------------------------------------------------------------------
1 | {
2 | "version": "0.2.0",
3 | "configurations": [
4 | {
5 | "name": "iOS Web",
6 | "type": "chrome",
7 | "request": "attach",
8 | "port": 9000,
9 | "url": "https://app.block-photos.com/*",
10 | "webRoot": "${workspaceRoot}/src"
11 | }
12 | ]
13 | }
14 |
--------------------------------------------------------------------------------
/nightwatch.json:
--------------------------------------------------------------------------------
1 | {
2 | "src_folders" : ["tests"],
3 | "output_folder" : "reports",
4 | "custom_commands_path" : "",
5 | "custom_assertions_path" : "",
6 | "page_objects_path" : "",
7 | "globals_path" : "",
8 |
9 | "selenium" : {
10 | "start_process" : false,
11 | "server_path" : "",
12 | "log_path" : "",
13 | "port" : 4444,
14 | "cli_args" : {
15 | "webdriver.chrome.driver" : "",
16 | "webdriver.gecko.driver" : "",
17 | "webdriver.edge.driver" : ""
18 | }
19 | },
20 |
21 | "test_settings" : {
22 | "default" : {
23 | "launch_url" : "https://localhost:9876",
24 | "selenium_port" : 4444,
25 | "selenium_host" : "localhost",
26 | "silent": true,
27 | "screenshots" : {
28 | "enabled" : false,
29 | "path" : ""
30 | },
31 | "desiredCapabilities": {
32 | "browserName": "chrome",
33 | "marionette": true
34 | }
35 | },
36 | "chrome" : {
37 | "desiredCapabilities": {
38 | "browserName": "chrome",
39 | "chromeOptions" : {
40 | "args" : ["--no-sandbox --user-data-dir=E:\\workspaces\\block-photos\\tests\\session"]
41 | }
42 | }
43 | },
44 |
45 | "edge" : {
46 | "desiredCapabilities": {
47 | "browserName": "MicrosoftEdge"
48 | }
49 | }
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "block-photos",
3 | "version": "4.2.0",
4 | "private": false,
5 | "description": "The app for storing your photos privatly in the cloud!",
6 | "repository": {
7 | "type": "git",
8 | "url": "https://github.com/nerdic-coder/block-photos"
9 | },
10 | "license": "MIT",
11 | "author": "Johan Axelsson",
12 | "main": "electron/main.js",
13 | "files": [
14 | "dist/"
15 | ],
16 | "scripts": {
17 | "build": "stencil build",
18 | "build:android": "npm run build && npx cap copy android",
19 | "build:ios": "npm run build && npx cap copy ios",
20 | "build:web": "npm run build",
21 | "electron": "electron-forge start",
22 | "electron-forge": "electron-forge",
23 | "make": "npm run build && npx cap copy electron && electron-forge make",
24 | "package": "npm run build && npx cap copy electron && electron-forge package",
25 | "publish": "npm run build && npx cap copy electron && electron-forge publish",
26 | "serve": "stencil build --dev --watch --serve",
27 | "start": "npm run build && npx cap copy electron && npm run electron",
28 | "test": "stencil test --spec --e2e",
29 | "test.watch": "stencil test --spec --e2e --watch",
30 | "e2e": "nightwatch",
31 | "selenium": "selenium-standalone install && selenium-standalone start",
32 | "format": "npx prettier --write src/**"
33 | },
34 | "config": {
35 | "forge": {
36 | "packagerConfig": {
37 | "name": "Block Photos",
38 | "icon": "electron/icons/mac/icon",
39 | "ignore": [
40 | ".vscode",
41 | "coverage",
42 | "android",
43 | "ios",
44 | "public",
45 | "www"
46 | ],
47 | "prune": true,
48 | "asar": true
49 | },
50 | "makers": [
51 | {
52 | "name": "@electron-forge/maker-dmg"
53 | },
54 | {
55 | "name": "@electron-forge/maker-squirrel",
56 | "config": {
57 | "name": "block_photos",
58 | "iconUrl": "https://raw.githubusercontent.com/nerdic-coder/block-photos/master/electron/icons/mac/icon.ico",
59 | "setupIcon": "electron/icons/mac/icon.ico"
60 | }
61 | },
62 | {
63 | "name": "@electron-forge/maker-zip"
64 | },
65 | {
66 | "name": "@electron-forge/maker-deb",
67 | "config": {}
68 | },
69 | {
70 | "name": "@electron-forge/maker-rpm",
71 | "config": {
72 | "options": {
73 | "maintainer": "Johan Axelsson",
74 | "homepage": "https://block-photos.com/"
75 | }
76 | }
77 | }
78 | ],
79 | "publishers": [
80 | {
81 | "name": "@electron-forge/publisher-github",
82 | "config": {
83 | "repository": {
84 | "owner": "nerdic-coder",
85 | "name": "block-photos"
86 | },
87 | "prerelease": true
88 | }
89 | }
90 | ],
91 | "github_repository": {
92 | "owner": "nerdic-coder",
93 | "name": "block-photos"
94 | }
95 | }
96 | },
97 | "dependencies": {
98 | "@capacitor/core": "1.0.0",
99 | "@ionic/core": "4.4.2",
100 | "@sentry/browser": "5.4.0",
101 | "blueimp-load-image": "2.21.0",
102 | "browser-image-compression": "1.0.5",
103 | "compressorjs": "1.0.5",
104 | "cors": "2.8.5",
105 | "electron-is-dev": "1.1.0",
106 | "electron-squirrel-startup": "^1.0.0",
107 | "is-electron": "^2.2.0",
108 | "js-file-downloader": "^1.1.0",
109 | "jszip": "3.2.1",
110 | "localforage": "1.7.3",
111 | "uuid": "3.3.2"
112 | },
113 | "devDependencies": {
114 | "@capacitor/android": "1.0.0",
115 | "@capacitor/cli": "1.0.0",
116 | "@capacitor/ios": "1.0.0",
117 | "@electron-forge/cli": "6.0.0-beta.39",
118 | "@electron-forge/maker-appx": "6.0.0-beta.39",
119 | "@electron-forge/maker-deb": "6.0.0-beta.39",
120 | "@electron-forge/maker-dmg": "6.0.0-beta.39",
121 | "@electron-forge/maker-pkg": "6.0.0-beta.39",
122 | "@electron-forge/maker-rpm": "6.0.0-beta.39",
123 | "@electron-forge/maker-squirrel": "6.0.0-beta.39",
124 | "@electron-forge/maker-zip": "6.0.0-beta.39",
125 | "@electron-forge/publisher-github": "6.0.0-beta.39",
126 | "@stencil/core": "0.18.1",
127 | "@types/jest": "24.0.13",
128 | "@types/puppeteer": "1.12.4",
129 | "blockstack": "19.2.1",
130 | "electron": "9.4.0",
131 | "husky": "2.4.0",
132 | "jest": "24.8.0",
133 | "jest-cli": "24.8.0",
134 | "nightwatch": "1.1.11",
135 | "prettier": "1.17.1",
136 | "pretty-quick": "1.11.0",
137 | "puppeteer": "1.17.0",
138 | "rollup-plugin-node-builtins": "2.1.2",
139 | "rollup-plugin-node-globals": "1.4.0",
140 | "selenium-standalone": "^6.16.0",
141 | "tslint": "5.17.0",
142 | "tslint-ionic-rules": "0.0.21",
143 | "workbox-build": "3.4.1"
144 | },
145 | "browserslist": [
146 | ">0.2%",
147 | "not dead",
148 | "not ie <= 11",
149 | "not op_mini all"
150 | ],
151 | "productName": "block-photos",
152 | "husky": {
153 | "hooks": {
154 | "pre-commit": "pretty-quick --staged"
155 | }
156 | }
157 | }
158 |
--------------------------------------------------------------------------------
/resources/btc-qrcode.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/resources/btc-qrcode.png
--------------------------------------------------------------------------------
/resources/etherium-qrcode.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/resources/etherium-qrcode.png
--------------------------------------------------------------------------------
/resources/iota-qrcode.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/resources/iota-qrcode.png
--------------------------------------------------------------------------------
/resources/xrp-qrcode.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/resources/xrp-qrcode.png
--------------------------------------------------------------------------------
/src/_headers:
--------------------------------------------------------------------------------
1 | ## A manifest.json:
2 | /manifest.json
3 | # Headers for that path:
4 | Access-Control-Allow-Origin: *
5 |
6 | ## A service-worker.js:
7 | /*service-worker.js
8 | # Headers for that path:
9 | Access-Control-Allow-Origin: *
10 | Content-Type: application/javascript; charset=UTF-8
11 |
12 | ## A precache-manifest:
13 | /*precache-manifest*
14 | # Headers for that path:
15 | Access-Control-Allow-Origin: *
16 | Content-Type: application/javascript; charset=UTF-8
17 |
--------------------------------------------------------------------------------
/src/_redirects:
--------------------------------------------------------------------------------
1 | /* /index.html 200
2 |
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-144x144.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-144x144.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-192x192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-192x192.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-256x256.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-256x256.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-36x36.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-36x36.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-384x384.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-384x384.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-48x48.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-48x48.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-512x512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-512x512.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-72x72.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-72x72.png
--------------------------------------------------------------------------------
/src/assets/icon/android-chrome-96x96.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/android-chrome-96x96.png
--------------------------------------------------------------------------------
/src/assets/icon/apple-touch-icon-120x120.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/apple-touch-icon-120x120.png
--------------------------------------------------------------------------------
/src/assets/icon/apple-touch-icon-152x152.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/apple-touch-icon-152x152.png
--------------------------------------------------------------------------------
/src/assets/icon/apple-touch-icon-180x180.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/apple-touch-icon-180x180.png
--------------------------------------------------------------------------------
/src/assets/icon/apple-touch-icon-60x60.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/apple-touch-icon-60x60.png
--------------------------------------------------------------------------------
/src/assets/icon/apple-touch-icon-76x76.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/apple-touch-icon-76x76.png
--------------------------------------------------------------------------------
/src/assets/icon/apple-touch-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/apple-touch-icon.png
--------------------------------------------------------------------------------
/src/assets/icon/favicon-1024x1024.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/favicon-1024x1024.png
--------------------------------------------------------------------------------
/src/assets/icon/favicon-16x16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/favicon-16x16.png
--------------------------------------------------------------------------------
/src/assets/icon/favicon-194x194.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/favicon-194x194.png
--------------------------------------------------------------------------------
/src/assets/icon/favicon-32x32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/favicon-32x32.png
--------------------------------------------------------------------------------
/src/assets/icon/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/favicon.ico
--------------------------------------------------------------------------------
/src/assets/icon/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/icon.png
--------------------------------------------------------------------------------
/src/assets/icon/mstile-144x144.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/mstile-144x144.png
--------------------------------------------------------------------------------
/src/assets/icon/mstile-150x150.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/mstile-150x150.png
--------------------------------------------------------------------------------
/src/assets/icon/mstile-310x150.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/mstile-310x150.png
--------------------------------------------------------------------------------
/src/assets/icon/mstile-310x310.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/mstile-310x310.png
--------------------------------------------------------------------------------
/src/assets/icon/mstile-70x70.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/icon/mstile-70x70.png
--------------------------------------------------------------------------------
/src/assets/icon/safari-pinned-tab.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/assets/placeholder-image.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/assets/placeholder-image.jpg
--------------------------------------------------------------------------------
/src/browserconfig.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 | #603cba
10 |
11 |
12 |
13 |
--------------------------------------------------------------------------------
/src/components.d.ts:
--------------------------------------------------------------------------------
1 | /* tslint:disable */
2 | /**
3 | * This is an autogenerated file created by the Stencil compiler.
4 | * It contains typing information for all components that exist in this project.
5 | */
6 |
7 |
8 | import '@stencil/core';
9 |
10 | import '@ionic/core';
11 | import 'ionicons';
12 | import {
13 | PhotoType,
14 | } from './models/photo-type';
15 |
16 |
17 | export namespace Components {
18 |
19 | interface BlockImg {
20 | 'photoId': string;
21 | 'phototType': PhotoType;
22 | 'refresh': number;
23 | 'rotate': boolean;
24 | }
25 | interface BlockImgAttributes extends StencilHTMLAttributes {
26 | 'photoId'?: string;
27 | 'phototType'?: PhotoType;
28 | 'refresh'?: number;
29 | 'rotate'?: boolean;
30 | }
31 |
32 | interface FilterPopover {
33 | 'selectedPhotos': any[];
34 | }
35 | interface FilterPopoverAttributes extends StencilHTMLAttributes {
36 | 'selectedPhotos'?: any[];
37 | }
38 |
39 | interface SelectAlbum {
40 | 'endCallback': any;
41 | 'selectedPhotos': any[];
42 | 'startCallback': any;
43 | }
44 | interface SelectAlbumAttributes extends StencilHTMLAttributes {
45 | 'endCallback'?: any;
46 | 'selectedPhotos'?: any[];
47 | 'startCallback'?: any;
48 | }
49 |
50 | interface AppAlbums {}
51 | interface AppAlbumsAttributes extends StencilHTMLAttributes {}
52 |
53 | interface AppPhoto {
54 | 'albumId': string;
55 | 'photoId': string;
56 | 'updateCallback': any;
57 | }
58 | interface AppPhotoAttributes extends StencilHTMLAttributes {
59 | 'albumId'?: string;
60 | 'photoId'?: string;
61 | 'updateCallback'?: any;
62 | }
63 |
64 | interface AppPhotos {
65 | 'albumId': string;
66 | 'photoId': string;
67 | }
68 | interface AppPhotosAttributes extends StencilHTMLAttributes {
69 | 'albumId'?: string;
70 | 'photoId'?: string;
71 | }
72 |
73 | interface AppRoot {}
74 | interface AppRootAttributes extends StencilHTMLAttributes {}
75 |
76 | interface AppSettings {}
77 | interface AppSettingsAttributes extends StencilHTMLAttributes {}
78 |
79 | interface AppSignin {}
80 | interface AppSigninAttributes extends StencilHTMLAttributes {}
81 | }
82 |
83 | declare global {
84 | interface StencilElementInterfaces {
85 | 'BlockImg': Components.BlockImg;
86 | 'FilterPopover': Components.FilterPopover;
87 | 'SelectAlbum': Components.SelectAlbum;
88 | 'AppAlbums': Components.AppAlbums;
89 | 'AppPhoto': Components.AppPhoto;
90 | 'AppPhotos': Components.AppPhotos;
91 | 'AppRoot': Components.AppRoot;
92 | 'AppSettings': Components.AppSettings;
93 | 'AppSignin': Components.AppSignin;
94 | }
95 |
96 | interface StencilIntrinsicElements {
97 | 'block-img': Components.BlockImgAttributes;
98 | 'filter-popover': Components.FilterPopoverAttributes;
99 | 'select-album': Components.SelectAlbumAttributes;
100 | 'app-albums': Components.AppAlbumsAttributes;
101 | 'app-photo': Components.AppPhotoAttributes;
102 | 'app-photos': Components.AppPhotosAttributes;
103 | 'app-root': Components.AppRootAttributes;
104 | 'app-settings': Components.AppSettingsAttributes;
105 | 'app-signin': Components.AppSigninAttributes;
106 | }
107 |
108 |
109 | interface HTMLBlockImgElement extends Components.BlockImg, HTMLStencilElement {}
110 | var HTMLBlockImgElement: {
111 | prototype: HTMLBlockImgElement;
112 | new (): HTMLBlockImgElement;
113 | };
114 |
115 | interface HTMLFilterPopoverElement extends Components.FilterPopover, HTMLStencilElement {}
116 | var HTMLFilterPopoverElement: {
117 | prototype: HTMLFilterPopoverElement;
118 | new (): HTMLFilterPopoverElement;
119 | };
120 |
121 | interface HTMLSelectAlbumElement extends Components.SelectAlbum, HTMLStencilElement {}
122 | var HTMLSelectAlbumElement: {
123 | prototype: HTMLSelectAlbumElement;
124 | new (): HTMLSelectAlbumElement;
125 | };
126 |
127 | interface HTMLAppAlbumsElement extends Components.AppAlbums, HTMLStencilElement {}
128 | var HTMLAppAlbumsElement: {
129 | prototype: HTMLAppAlbumsElement;
130 | new (): HTMLAppAlbumsElement;
131 | };
132 |
133 | interface HTMLAppPhotoElement extends Components.AppPhoto, HTMLStencilElement {}
134 | var HTMLAppPhotoElement: {
135 | prototype: HTMLAppPhotoElement;
136 | new (): HTMLAppPhotoElement;
137 | };
138 |
139 | interface HTMLAppPhotosElement extends Components.AppPhotos, HTMLStencilElement {}
140 | var HTMLAppPhotosElement: {
141 | prototype: HTMLAppPhotosElement;
142 | new (): HTMLAppPhotosElement;
143 | };
144 |
145 | interface HTMLAppRootElement extends Components.AppRoot, HTMLStencilElement {}
146 | var HTMLAppRootElement: {
147 | prototype: HTMLAppRootElement;
148 | new (): HTMLAppRootElement;
149 | };
150 |
151 | interface HTMLAppSettingsElement extends Components.AppSettings, HTMLStencilElement {}
152 | var HTMLAppSettingsElement: {
153 | prototype: HTMLAppSettingsElement;
154 | new (): HTMLAppSettingsElement;
155 | };
156 |
157 | interface HTMLAppSigninElement extends Components.AppSignin, HTMLStencilElement {}
158 | var HTMLAppSigninElement: {
159 | prototype: HTMLAppSigninElement;
160 | new (): HTMLAppSigninElement;
161 | };
162 |
163 | interface HTMLElementTagNameMap {
164 | 'block-img': HTMLBlockImgElement
165 | 'filter-popover': HTMLFilterPopoverElement
166 | 'select-album': HTMLSelectAlbumElement
167 | 'app-albums': HTMLAppAlbumsElement
168 | 'app-photo': HTMLAppPhotoElement
169 | 'app-photos': HTMLAppPhotosElement
170 | 'app-root': HTMLAppRootElement
171 | 'app-settings': HTMLAppSettingsElement
172 | 'app-signin': HTMLAppSigninElement
173 | }
174 |
175 | interface ElementTagNameMap {
176 | 'block-img': HTMLBlockImgElement;
177 | 'filter-popover': HTMLFilterPopoverElement;
178 | 'select-album': HTMLSelectAlbumElement;
179 | 'app-albums': HTMLAppAlbumsElement;
180 | 'app-photo': HTMLAppPhotoElement;
181 | 'app-photos': HTMLAppPhotosElement;
182 | 'app-root': HTMLAppRootElement;
183 | 'app-settings': HTMLAppSettingsElement;
184 | 'app-signin': HTMLAppSigninElement;
185 | }
186 |
187 |
188 | export namespace JSX {
189 | export interface Element {}
190 | export interface IntrinsicElements extends StencilIntrinsicElements {
191 | [tagName: string]: any;
192 | }
193 | }
194 | export interface HTMLAttributes extends StencilHTMLAttributes {}
195 |
196 | }
197 |
--------------------------------------------------------------------------------
/src/components/block-img.spec.ts:
--------------------------------------------------------------------------------
1 | // import BlockImg from './block-img';
2 |
3 | import blockstack from 'blockstack';
4 |
5 | jest.mock('blockstack');
6 |
7 | it('renders without crashing', () => {
8 | const mockResponse = 'base64mumbojumbo';
9 | blockstack.getFile.mockReturnValue(Promise.resolve(mockResponse));
10 |
11 | // const div = document.createElement('div');
12 | // ReactDOM.render(
13 | //
14 | // , div);
15 | // ReactDOM.unmountComponentAtNode(div);
16 | });
17 |
--------------------------------------------------------------------------------
/src/components/block-img.tsx:
--------------------------------------------------------------------------------
1 | import { Component, Prop, State, Watch } from '@stencil/core';
2 | import loadImage from 'blueimp-load-image';
3 |
4 | import PhotosService from '../services/photos-service';
5 | import { PhotoType } from '../models/photo-type';
6 |
7 | @Component({
8 | tag: 'block-img'
9 | })
10 | export class BlockImg {
11 | @Prop() photoId: string;
12 | @Prop() rotate: boolean;
13 | @Prop() refresh: number;
14 | @Prop() phototType: PhotoType = PhotoType.Download;
15 |
16 | @State() source: string;
17 | @State() isLoaded: boolean;
18 | @State() rotation: number;
19 |
20 | constructor() {
21 | this.rotation = 1;
22 | }
23 |
24 | componentWillLoad(): void {
25 | this.getPhoto();
26 | }
27 |
28 | @Watch('photoId')
29 | photoIdDidUpdate(newValue: string, oldValue: string): void {
30 | if (newValue !== oldValue) {
31 | this.getPhoto();
32 | }
33 | }
34 |
35 | @Watch('refresh')
36 | refreshDidUpdate(newValue: number, oldValue: number): void {
37 | if (newValue !== oldValue) {
38 | this.getPhoto(newValue);
39 | }
40 | }
41 |
42 | async getPhoto(newRotation?: number): Promise {
43 | const { photoId, rotate } = this;
44 | let rotation = 1;
45 |
46 | if (photoId === null) {
47 | this.source =
48 | '';
49 | return;
50 | }
51 |
52 | const metadata: PhotoMetadata = await PhotosService.getPhotoMetaData(
53 | photoId
54 | );
55 | const photo = await PhotosService.loadPhoto(
56 | metadata,
57 | this.phototType,
58 | true
59 | );
60 | let base64;
61 | if (photo) {
62 | base64 = photo.base64;
63 | }
64 | if (newRotation) {
65 | rotation = newRotation;
66 | } else if (
67 | metadata &&
68 | metadata.stats &&
69 | metadata.stats.exifdata &&
70 | metadata.stats.exifdata.tags.Orientation
71 | ) {
72 | rotation = metadata.stats.exifdata.tags.Orientation;
73 | // Handle correct orientation for iOS
74 | if (
75 | this.iOS() &&
76 | photo.phototType === PhotoType.Download &&
77 | metadata.stats.exifdata.tags.OriginalOrientation
78 | ) {
79 | const originalOrientation =
80 | metadata.stats.exifdata.tags.OriginalOrientation;
81 | // If the orientation is unchanged don't rotate at all with CSS, iOS handles it automatic
82 | if (rotation === originalOrientation) {
83 | rotation = 1;
84 | } else if (rotation === 1 && originalOrientation === 6) {
85 | rotation = 8;
86 | } else if (rotation === 1) {
87 | rotation = originalOrientation;
88 | } else if (rotation === 3 && originalOrientation === 6) {
89 | rotation = 6;
90 | } else if (rotation === 8 && originalOrientation === 6) {
91 | rotation = 3;
92 | } else if (rotation === 3 && originalOrientation === 8) {
93 | rotation = 6;
94 | } else if (rotation === 6 && originalOrientation === 8) {
95 | rotation = 3;
96 | } else if (rotation === 8 && originalOrientation === 3) {
97 | rotation = 6;
98 | } else if (rotation === 6 && originalOrientation === 3) {
99 | rotation = 8;
100 | }
101 | }
102 | }
103 |
104 | // Set photo orientation from exif
105 | if (rotate) {
106 | const imageOptions = {
107 | orientation: rotation
108 | };
109 | loadImage(
110 | base64,
111 | processedPhoto => {
112 | this.handleProcessedPhoto(processedPhoto);
113 | },
114 | imageOptions
115 | );
116 | } else {
117 | this.rotation = rotation;
118 | this.source = base64;
119 | }
120 | }
121 |
122 | handleProcessedPhoto(processedPhoto: any): void {
123 | if (processedPhoto.type === 'error') {
124 | // TODO: show error message
125 | } else if (processedPhoto.tagName === 'CANVAS') {
126 | this.source = processedPhoto.toDataURL();
127 | } else {
128 | this.source = processedPhoto.src;
129 | }
130 | }
131 |
132 | iOS(): boolean {
133 | const iDevices = [
134 | 'iPad Simulator',
135 | 'iPhone Simulator',
136 | 'iPod Simulator',
137 | 'iPad',
138 | 'iPhone',
139 | 'iPod'
140 | ];
141 |
142 | if (navigator.platform) {
143 | while (iDevices.length) {
144 | if (navigator.platform === iDevices.pop()) {
145 | return true;
146 | }
147 | }
148 | }
149 |
150 | return false;
151 | }
152 |
153 | preventDrag(event: any): boolean {
154 | event.preventDefault();
155 | return false;
156 | }
157 |
158 | photoLoaded() {
159 | this.isLoaded = true;
160 | }
161 |
162 | render() {
163 | const { isLoaded, source, rotation } = this;
164 |
165 | return [
166 | this.preventDrag(event)}
172 | onIonImgDidLoad={() => this.photoLoaded()}
173 | />,
174 |
179 | ];
180 | }
181 | }
182 |
--------------------------------------------------------------------------------
/src/components/filter-popover.tsx:
--------------------------------------------------------------------------------
1 | import { Component, Prop } from '@stencil/core';
2 | import { RangeChangeEventDetail, RangeValue } from '@ionic/core';
3 |
4 | declare var Caman;
5 | // import PresentingService from '../services/presenting-service';
6 | // import PhotosService from '../services/photos-service';
7 |
8 | @Component({
9 | tag: 'filter-popover'
10 | })
11 | export class FilterPopover {
12 | @Prop() selectedPhotos: any[] = [];
13 |
14 | private brightness: RangeValue = 0;
15 | private contrast: RangeValue = 0;
16 | private saturation: RangeValue = 0;
17 |
18 | async closePopover() {
19 | const popoverController = document.querySelector('ion-popover-controller');
20 | await popoverController.componentOnReady();
21 | popoverController.dismiss();
22 | this.selectedPhotos = null;
23 | }
24 |
25 | setBrightness(event: CustomEvent) {
26 | console.log(event.detail.value);
27 | this.brightness = event.detail.value;
28 | this.applyFilters();
29 | }
30 |
31 | setSaturation(event: CustomEvent) {
32 | console.log(event.detail.value);
33 | this.saturation = event.detail.value;
34 | this.applyFilters();
35 | }
36 |
37 | setContrast(event: CustomEvent) {
38 | console.log(event.detail.value);
39 | this.contrast = event.detail.value;
40 | this.applyFilters();
41 | }
42 |
43 | applyFilters() {
44 | const brightness = this.brightness;
45 | const contrast = this.contrast;
46 | const saturation = this.saturation;
47 | // const photoId = this.selectedPhotos[0];
48 |
49 | Caman('#img-' + this.selectedPhotos[0], function() {
50 | // this.greyscale();
51 | this.revert(false);
52 | this.brightness(brightness);
53 | this.contrast(contrast);
54 | this.saturation(saturation);
55 | this.render(async () => {
56 | // const result = await PhotosService.updatePhoto(photoId, this.toBase64());
57 | // console.log('Caman result ', result);
58 | });
59 | });
60 | }
61 |
62 | render() {
63 | return (
64 |
65 |
66 | Filters
67 |
68 |
69 |
70 | Brightness
71 |
72 |
73 | this.setBrightness(event)}
75 | min={-100}
76 | max={100}
77 | color="secondary"
78 | pin={true}
79 | >
80 | -100
81 | 100
82 |
83 |
84 |
85 |
86 | Contrast
87 |
88 |
89 | this.setContrast(event)}
91 | min={-100}
92 | max={100}
93 | color="secondary"
94 | pin={true}
95 | >
96 | -100
97 | 100
98 |
99 |
100 |
101 |
102 | Saturation
103 |
104 |
105 | this.setSaturation(event)}
107 | min={-100}
108 | max={100}
109 | color="secondary"
110 | pin={true}
111 | >
112 | -100
113 | 100
114 |
115 |
116 |
117 | );
118 | }
119 | }
120 |
--------------------------------------------------------------------------------
/src/components/select-album.tsx:
--------------------------------------------------------------------------------
1 | import { Component, Prop, State } from '@stencil/core';
2 |
3 | import AlbumsService from '../services/albums-service';
4 | import PresentingService from '../services/presenting-service';
5 | import PhotosService from '../services/photos-service';
6 |
7 | @Component({
8 | tag: 'select-album'
9 | })
10 | export class SelectAlbum {
11 | private present: PresentingService;
12 |
13 | @State() albums: any[] = [];
14 | @State() isLoaded: boolean;
15 |
16 | @Prop() selectedPhotos: any[] = [];
17 | @Prop() startCallback: any;
18 | @Prop() endCallback: any;
19 |
20 | constructor() {
21 | this.present = new PresentingService();
22 | }
23 |
24 | componentWillLoad() {
25 | this.isLoaded = false;
26 | }
27 |
28 | async componentDidLoad() {
29 | this.loadAlbums(false);
30 | }
31 |
32 | async loadAlbums(sync?: boolean) {
33 | try {
34 | const albumsResponse = await AlbumsService.getAlbums(sync);
35 | this.albums = albumsResponse.albums;
36 |
37 | this.isLoaded = true;
38 |
39 | this.handleAlbumErrors(albumsResponse);
40 | } catch (error) {
41 | this.isLoaded = true;
42 | }
43 | }
44 |
45 | private handleAlbumErrors(albumsResponse: any) {
46 | if (albumsResponse.errorsList && albumsResponse.errorsList.length > 0) {
47 | for (const error of albumsResponse.errorsList) {
48 | if (error.errorCode === 'err_cache') {
49 | this.present.toast('Failed to load local albums. Please try again!');
50 | } else if (error.errorCode) {
51 | this.present.toast(
52 | 'Could not load albums from blockstack. Please try again or create some albums if you have none!'
53 | );
54 | }
55 | }
56 | }
57 | }
58 |
59 | async addPhotosToAlbum(event: any, album: any) {
60 | event.preventDefault();
61 | await this.closePopover();
62 | if (this.startCallback) {
63 | this.startCallback();
64 | }
65 | const result: boolean = await PhotosService.addPhotosToAlbum(
66 | album.albumId,
67 | this.selectedPhotos
68 | );
69 | if (result) {
70 | this.present.toast('Photo(s) added to album "' + album.albumName + '".');
71 | } else {
72 | this.present.toast(
73 | 'Failed to add photo(s) to album "' + album.albumName + '".'
74 | );
75 | }
76 | if (this.endCallback) {
77 | this.endCallback();
78 | }
79 | }
80 |
81 | async closePopover() {
82 | const popoverController = document.querySelector('ion-popover-controller');
83 | await popoverController.componentOnReady();
84 | popoverController.dismiss();
85 | this.selectedPhotos = null;
86 | }
87 |
88 | render() {
89 | if (this.isLoaded) {
90 | return (
91 |
92 |
93 | Add to album
94 |
95 | {this.albums.map(album => (
96 | this.addPhotosToAlbum(event, album)}
100 | >
101 | {album.albumName}
102 |
103 | ))}
104 |
105 | );
106 | } else {
107 | return ;
108 | }
109 | }
110 | }
111 |
--------------------------------------------------------------------------------
/src/global/app.css:
--------------------------------------------------------------------------------
1 | /* Document Level Styles */
2 |
3 | /*
4 | The imports below are needed to include our light dom css for global styles such as fonts and colors.
5 | You can comment out any of these imports if you do not need that css. For example, if you have your own
6 | global font family css then you can comment out the typography.css import.
7 | */
8 |
9 | /** Core CSS required for ionic components to work property */
10 | @import '~@ionic/core/css/core.css';
11 |
12 | /** Basic CSS for apps built with Ionic */
13 | @import '~@ionic/core/css/normalize.css';
14 | @import '~@ionic/core/css/structure.css';
15 | @import '~@ionic/core/css/typography.css';
16 | @import '~@ionic/core/css/display.css';
17 |
18 | /** Optional CSS utils that can be commented out */
19 | @import '~@ionic/core/css/padding.css';
20 | @import '~@ionic/core/css/float-elements.css';
21 | @import '~@ionic/core/css/text-alignment.css';
22 | @import '~@ionic/core/css/text-transformation.css';
23 | @import '~@ionic/core/css/flex-utils.css';
24 |
25 | /*
26 | The CSS Variables below can be used to theme your app.
27 | For more info on CSS variables check out:
28 | https://developer.mozilla.org/en-US/docs/Web/CSS/Using_CSS_variables
29 |
30 | More info about color theming using Ionic:
31 | https://beta.ionicframework.com/docs/theming/color-generator
32 | */
33 | body {
34 | margin: 0;
35 | padding: 0;
36 | font-family: sans-serif;
37 | }
38 |
39 | :root {
40 | --ion-color-primary: #220631;
41 | --ion-color-primary-rgb: 34, 6, 49;
42 | --ion-color-primary-contrast: #ffffff;
43 | --ion-color-primary-contrast-rgb: 255, 255, 255;
44 | --ion-color-primary-shade: #1e052b;
45 | --ion-color-primary-tint: #381f46;
46 |
47 | --ion-color-secondary: #663399;
48 | --ion-color-secondary-rgb: 12, 209, 232;
49 | --ion-color-secondary-contrast: #ffffff;
50 | --ion-color-secondary-contrast-rgb: 255, 255, 255;
51 | --ion-color-secondary-shade: #9249db;
52 | --ion-color-secondary-tint: rgb(129, 40, 218);
53 |
54 | --ion-color-tertiary: #7044ff;
55 | --ion-color-tertiary-rgb: 112, 68, 255;
56 | --ion-color-tertiary-contrast: #ffffff;
57 | --ion-color-tertiary-contrast-rgb: 255, 255, 255;
58 | --ion-color-tertiary-shade: #633ce0;
59 | --ion-color-tertiary-tint: #7e57ff;
60 |
61 | --ion-color-success: #10dc60;
62 | --ion-color-success-rgb: 16, 220, 96;
63 | --ion-color-success-contrast: #ffffff;
64 | --ion-color-success-contrast-rgb: 255, 255, 255;
65 | --ion-color-success-shade: #0ec254;
66 | --ion-color-success-tint: #28e070;
67 |
68 | --ion-color-warning: #ffce00;
69 | --ion-color-warning-rgb: 255, 206, 0;
70 | --ion-color-warning-contrast: #ffffff;
71 | --ion-color-warning-contrast-rgb: 255, 255, 255;
72 | --ion-color-warning-shade: #e0b500;
73 | --ion-color-warning-tint: #ffd31a;
74 |
75 | --ion-color-danger: #8a0000;
76 | --ion-color-danger-rgb: 245, 61, 61;
77 | --ion-color-danger-contrast: #ffffff;
78 | --ion-color-danger-contrast-rgb: 255, 255, 255;
79 | --ion-color-danger-shade: #d33939;
80 | --ion-color-danger-tint: #f25454;
81 |
82 | --ion-color-dark: #000;
83 | --ion-color-dark-rgb: 34, 34, 34;
84 | --ion-color-dark-contrast: #ffffff;
85 | --ion-color-dark-contrast-rgb: 255, 255, 255;
86 | --ion-color-dark-shade: #1e2023;
87 | --ion-color-dark-tint: #383a3e;
88 |
89 | --ion-color-medium: #cbcbcb;
90 | --ion-color-medium-rgb: 203, 203, 203;
91 | --ion-color-medium-contrast: #000000;
92 | --ion-color-medium-contrast-rgb: 0, 0, 0;
93 | --ion-color-medium-shade: #b3b3b3;
94 | --ion-color-medium-tint: #d0d0d0;
95 |
96 | --ion-color-light: #f4f5f8;
97 | --ion-color-light-rgb: 244, 244, 244;
98 | --ion-color-light-contrast: #000000;
99 | --ion-color-light-contrast-rgb: 0, 0, 0;
100 | --ion-color-light-shade: #d7d8da;
101 | --ion-color-light-tint: #f5f6f9;
102 | }
103 |
104 | ion-spinner {
105 | display: block;
106 | margin: auto;
107 | }
108 |
109 | .pointer {
110 | cursor: pointer;
111 | }
112 |
113 | .square,
114 | .full {
115 | width: 100%;
116 | height: 100%;
117 | }
118 |
119 | .square:after {
120 | content: '';
121 | display: block;
122 | padding-bottom: 100%;
123 | }
124 |
125 | .square ion-img {
126 | position: absolute;
127 | object-fit: cover;
128 | width: 100%;
129 | height: 100%;
130 | padding: 2px;
131 | user-select: none;
132 | -moz-user-select: none;
133 | -webkit-user-drag: none;
134 | -webkit-user-select: none;
135 | -ms-user-select: none;
136 | }
137 |
138 | .selected .square ion-img {
139 | opacity: 0.7;
140 | }
141 |
142 | .selected .square {
143 | background-color: var(--ion-color-secondary);
144 | }
145 |
146 | .floatInput {
147 | position: absolute;
148 | top: 6px;
149 | left: 6px;
150 | }
151 |
152 | .floatIcon {
153 | position: absolute;
154 | top: 0;
155 | left: 4px;
156 | }
157 |
158 | .square ion-spinner,
159 | ion-slide ion-spinner {
160 | position: absolute;
161 | left: 45%;
162 | top: 45%;
163 | object-fit: cover;
164 | margin: auto;
165 | padding: 2px;
166 | }
167 |
168 | .square .rotation-6 {
169 | transform: rotate(90deg);
170 | }
171 |
172 | .square .rotation-3 {
173 | transform: rotate(180deg);
174 | }
175 |
176 | .square .rotation-8 {
177 | transform: rotate(270deg);
178 | }
179 |
180 | .hidden {
181 | visibility: hidden;
182 | }
183 |
184 | #file-upload {
185 | display: none;
186 | }
187 |
188 | .signin,
189 | .profile {
190 | --background: linear-gradient(
191 | 135deg,
192 | rgba(60, 8, 118, 0.8) 0%,
193 | rgba(250, 0, 118, 0.8) 100%
194 | );
195 | }
196 |
197 | .unselectable {
198 | -webkit-touch-callout: none;
199 | -webkit-user-select: none;
200 | -khtml-user-select: none;
201 | -moz-user-select: none;
202 | -ms-user-select: none;
203 | user-select: none;
204 | }
205 |
206 | .router-modal,
207 | .router-modal .modal-wrapper {
208 | width: 100%;
209 | height: 100%;
210 | border: 0 solid #000;
211 | }
212 |
213 | ion-content {
214 | --padding-end: env(safe-area-inset-right);
215 | --padding-start: env(safe-area-inset-left);
216 | --padding-bottom: env(safe-area-inset-bottom);
217 | }
218 |
219 | ion-card ion-item {
220 | --min-height: 19px;
221 | }
222 |
223 | .loadingToolbar {
224 | display: none;
225 | }
226 |
227 | .loadingToolbar.show {
228 | display: block;
229 | }
230 |
--------------------------------------------------------------------------------
/src/global/app.ts:
--------------------------------------------------------------------------------
1 | import '@ionic/core';
2 |
3 | // import { setupConfig } from '@ionic/core';
4 |
5 | // setupConfig({
6 | // mode: 'ios'
7 | // });
8 |
--------------------------------------------------------------------------------
/src/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Block Photos
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
49 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
107 |
108 |
109 |
110 |
111 |
112 |
113 |
114 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
122 |
--------------------------------------------------------------------------------
/src/interfaces.d.ts:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/interfaces.d.ts
--------------------------------------------------------------------------------
/src/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "block-photos",
3 | "short_name": "block-photos",
4 | "start_url": ".",
5 | "display": "standalone",
6 | "icons": [
7 | {
8 | "src": "https://app.block-photos.com/assets/icon/icon.png",
9 | "sizes": "512x512",
10 | "type": "image/png"
11 | }
12 | ],
13 | "background_color": "#488aff",
14 | "theme_color": "#488aff"
15 | }
16 |
--------------------------------------------------------------------------------
/src/models/photo-metadata.ts:
--------------------------------------------------------------------------------
1 | // The metadata stored for a photo
2 | interface PhotoMetadata {
3 | id: string;
4 | filename: string;
5 | stats: any;
6 | type: string;
7 | size: number;
8 | uploadedDate: Date;
9 | albums: string[];
10 | }
11 |
--------------------------------------------------------------------------------
/src/models/photo-type.ts:
--------------------------------------------------------------------------------
1 | // The iamge format to be displayed or fetched
2 | export enum PhotoType {
3 | // The original high res image
4 | Download,
5 | // For fullscreen viewing in app
6 | Viewer,
7 | // For lists of photos
8 | Thumbnail
9 | }
10 |
--------------------------------------------------------------------------------
/src/pages/app-albums/app-albums.e2e.ts:
--------------------------------------------------------------------------------
1 | import { newE2EPage } from '@stencil/core/testing';
2 |
3 | describe('app-profile', () => {
4 | it('renders', async () => {
5 | const page = await newE2EPage();
6 | await page.setContent(' ');
7 |
8 | const element = await page.find('app-profile');
9 | expect(element).toHaveClass('hydrated');
10 | });
11 |
12 | it('displays the specified name', async () => {
13 | const page = await newE2EPage({ url: '/profile/joseph' });
14 |
15 | const element = await page.find('app-profile ion-content p');
16 | expect(element.textContent).toContain('My name is Joseph.');
17 | });
18 | });
19 |
--------------------------------------------------------------------------------
/src/pages/app-albums/app-albums.spec.ts:
--------------------------------------------------------------------------------
1 | import { AppProfile } from './app-profile';
2 |
3 | describe('app-profile', () => {
4 | it('builds', () => {
5 | expect(new AppProfile()).toBeTruthy();
6 | });
7 |
8 | describe('normalization', () => {
9 | it('returns a blank string if the name is undefined', () => {
10 | const component = new AppProfile();
11 | expect(component.formattedName()).toEqual('');
12 | });
13 |
14 | it('capitalizes the first letter', () => {
15 | const component = new AppProfile();
16 | component.name = 'quincy';
17 | expect(component.formattedName()).toEqual('Quincy');
18 | });
19 |
20 | it('lower-cases the following letters', () => {
21 | const component = new AppProfile();
22 | component.name = 'JOSEPH';
23 | expect(component.formattedName()).toEqual('Joseph');
24 | });
25 |
26 | it('handles single letter names', () => {
27 | const component = new AppProfile();
28 | component.name = 'q';
29 | expect(component.formattedName()).toEqual('Q');
30 | });
31 | });
32 | });
33 |
--------------------------------------------------------------------------------
/src/pages/app-photo/app-photo.css:
--------------------------------------------------------------------------------
1 | app-photo ion-slides {
2 | height: 100%;
3 | }
4 |
5 | app-photo .swiper-slide img {
6 | width: 100%;
7 | }
8 |
--------------------------------------------------------------------------------
/src/pages/app-photo/app-photo.e2e.ts:
--------------------------------------------------------------------------------
1 | import { newE2EPage } from '@stencil/core/testing';
2 |
3 | describe('app-photo', () => {
4 | it('renders', async () => {
5 | const page = await newE2EPage();
6 | await page.setContent(' ');
7 |
8 | const element = await page.find('app-photo');
9 | expect(element).toHaveClass('hydrated');
10 | });
11 |
12 | it('displays the specified name', async () => {
13 | const page = await newE2EPage({ url: '/photos/joseph' });
14 |
15 | const element = await page.find('app-photo ion-content p');
16 | expect(element.textContent).toContain('My name is Joseph.');
17 | });
18 | });
19 |
--------------------------------------------------------------------------------
/src/pages/app-photo/app-photo.spec.ts:
--------------------------------------------------------------------------------
1 | import { AppPhoto } from './app-photo';
2 |
3 | describe('app-photo', () => {
4 | it('builds', () => {
5 | expect(new AppPhoto()).toBeTruthy();
6 | });
7 |
8 | describe('normalization', () => {
9 | it('returns a blank string if the name is undefined', () => {
10 | const component = new AppPhoto();
11 | expect(component.formattedName()).toEqual('');
12 | });
13 |
14 | it('capitalizes the first letter', () => {
15 | const component = new AppPhoto();
16 | component.name = 'quincy';
17 | expect(component.formattedName()).toEqual('Quincy');
18 | });
19 |
20 | it('lower-cases the following letters', () => {
21 | const component = new AppPhoto();
22 | component.name = 'JOSEPH';
23 | expect(component.formattedName()).toEqual('Joseph');
24 | });
25 |
26 | it('handles single letter names', () => {
27 | const component = new AppPhoto();
28 | component.name = 'q';
29 | expect(component.formattedName()).toEqual('Q');
30 | });
31 | });
32 | });
33 |
--------------------------------------------------------------------------------
/src/pages/app-photos/app-photos.css:
--------------------------------------------------------------------------------
1 | .upload-grid,
2 | .upload-grid ion-row {
3 | height: 100%;
4 | max-height: 100%;
5 | }
6 |
7 | .cloud-icon {
8 | width: 88px;
9 | height: 88px;
10 | }
11 |
12 | @media only screen and (max-width: 600px) {
13 | ion-col h3 {
14 | font-size: 14px;
15 | }
16 |
17 | .cloud-icon {
18 | width: 28px;
19 | height: 28px;
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/src/pages/app-photos/app-photos.e2e.ts:
--------------------------------------------------------------------------------
1 | import { newE2EPage } from '@stencil/core/testing';
2 |
3 | describe('app-photos', () => {
4 | it('renders', async () => {
5 | const page = await newE2EPage();
6 | await page.setContent(' ');
7 |
8 | const element = await page.find('app-photos');
9 | expect(element).toHaveClass('hydrated');
10 | });
11 |
12 | it('displays the specified name', async () => {
13 | const page = await newE2EPage({ url: '/photos/joseph' });
14 |
15 | const element = await page.find('app-photos ion-content p');
16 | expect(element.textContent).toContain('My name is Joseph.');
17 | });
18 | });
19 |
--------------------------------------------------------------------------------
/src/pages/app-photos/app-photos.spec.ts:
--------------------------------------------------------------------------------
1 | import { AppPhotos } from './app-photos';
2 |
3 | describe('app-photos', () => {
4 | it('builds', () => {
5 | expect(new AppPhotos()).toBeTruthy();
6 | });
7 |
8 | describe('normalization', () => {
9 | it('returns a blank string if the name is undefined', () => {
10 | const component = new AppPhotos();
11 | expect(component.formattedName()).toEqual('');
12 | });
13 |
14 | it('capitalizes the first letter', () => {
15 | const component = new AppPhotos();
16 | component.name = 'quincy';
17 | expect(component.formattedName()).toEqual('Quincy');
18 | });
19 |
20 | it('lower-cases the following letters', () => {
21 | const component = new AppPhotos();
22 | component.name = 'JOSEPH';
23 | expect(component.formattedName()).toEqual('Joseph');
24 | });
25 |
26 | it('handles single letter names', () => {
27 | const component = new AppPhotos();
28 | component.name = 'q';
29 | expect(component.formattedName()).toEqual('Q');
30 | });
31 | });
32 | });
33 |
--------------------------------------------------------------------------------
/src/pages/app-root/app-root.css:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/nerdic-coder/block-photos/a6d76f1b3d0a32cf0d6c458fad255462cb367f6e/src/pages/app-root/app-root.css
--------------------------------------------------------------------------------
/src/pages/app-root/app-root.e2e.ts:
--------------------------------------------------------------------------------
1 | import { newE2EPage } from '@stencil/core/testing';
2 |
3 | describe('app-root', () => {
4 | it('renders', async () => {
5 | const page = await newE2EPage({ url: '/' });
6 |
7 | const element = await page.find('app-root');
8 | expect(element).toHaveClass('hydrated');
9 | });
10 |
11 | it('renders an ion-app', async () => {
12 | const page = await newE2EPage({ url: '/' });
13 |
14 | const element = await page.find('app-root > ion-app');
15 | expect(element).toHaveClass('hydrated');
16 | });
17 | });
18 |
--------------------------------------------------------------------------------
/src/pages/app-root/app-root.spec.ts:
--------------------------------------------------------------------------------
1 | import { AppRoot } from './app-root';
2 |
3 | describe('app-root', () => {
4 | it('builds', () => {
5 | expect(new AppRoot()).toBeTruthy();
6 | });
7 |
8 | describe('onSWUpdate', () => {
9 | let appRoot: AppRoot;
10 | let mockToast;
11 | beforeEach(() => {
12 | appRoot = new AppRoot();
13 | mockToast = {
14 | present: jest.fn().mockReturnValue(Promise.resolve()),
15 | onWillDismiss: jest.fn().mockReturnValue(Promise.resolve())
16 | };
17 | appRoot.toastCtrl = {
18 | create: jest.fn().mockReturnValue(Promise.resolve(mockToast))
19 | } as any;
20 | window.location.reload = jest.fn();
21 | });
22 |
23 | it('creates a new toast', () => {
24 | appRoot.onSWUpdate();
25 | expect(appRoot.toastCtrl.create).toHaveBeenCalledTimes(1);
26 | expect(appRoot.toastCtrl.create).toHaveBeenCalledWith({
27 | message: 'New version available',
28 | showCloseButton: true,
29 | closeButtonText: 'Reload'
30 | });
31 | });
32 |
33 | it('presents the toast', async () => {
34 | await appRoot.onSWUpdate();
35 | expect(mockToast.present).toHaveReturnedTimes(1);
36 | });
37 |
38 | it('reloads the app', async () => {
39 | await appRoot.onSWUpdate();
40 | expect(window.location.reload).toHaveReturnedTimes(1);
41 | });
42 | });
43 | });
44 |
--------------------------------------------------------------------------------
/src/pages/app-root/app-root.tsx:
--------------------------------------------------------------------------------
1 | import { Component, Listen, Prop, State } from '@stencil/core';
2 | import { Plugins } from '@capacitor/core';
3 | import * as Sentry from '@sentry/browser';
4 | import localForage from 'localforage';
5 |
6 | import AnalyticsService from '../../services/analytics-service';
7 | import CacheService from '../../services/cache-service';
8 | import SettingsService from '../../services/settings-service';
9 |
10 | declare var blockstack;
11 |
12 | @Component({
13 | tag: 'app-root',
14 | styleUrl: 'app-root.css'
15 | })
16 | export class AppRoot {
17 | @Prop({ connect: 'ion-toast-controller' })
18 | toastCtrl: HTMLIonToastControllerElement;
19 |
20 | @State() isAuthenticated: boolean;
21 |
22 | /**
23 | * Handle service worker updates correctly.
24 | * This code will show a toast letting the
25 | * user of the PWA know that there is a
26 | * new version available. When they click the
27 | * reload button it then reloads the page
28 | * so that the new service worker can take over
29 | * and serve the fresh content
30 | */
31 | @Listen('window:swUpdate')
32 | async onSWUpdate() {
33 | const toast = await this.toastCtrl.create({
34 | message: 'New version available',
35 | showCloseButton: true,
36 | closeButtonText: 'Reload'
37 | });
38 | await toast.present();
39 | await toast.onWillDismiss();
40 | window.location.reload();
41 | }
42 |
43 | async componentWillLoad() {
44 | localForage.config({
45 | name: 'BlockPhotos',
46 | version: 1.0,
47 | storeName: 'blockphotos', // Should be alphanumeric, with underscores.
48 | description: 'Block Photos Cache'
49 | });
50 | try {
51 | if (await SettingsService.getAnalyticsSetting(true)) {
52 | Sentry.init({
53 | dsn: 'https://2b0b525209b646f49e438cff86c3e117@sentry.io/1331915',
54 | release: 'block-photos@4.2'
55 | });
56 | }
57 | } catch (error) {
58 | Sentry.init({
59 | dsn: 'https://2b0b525209b646f49e438cff86c3e117@sentry.io/1331915',
60 | release: 'block-photos@4.2'
61 | });
62 | }
63 |
64 | this.initCapacitor();
65 |
66 | const userSession = new blockstack.UserSession();
67 | this.isAuthenticated = userSession.isUserSignedIn();
68 | }
69 |
70 | async componentDidLoad() {
71 | const router: any = document.querySelector('ion-router');
72 | await router.componentOnReady();
73 | router.addEventListener('ionRouteDidChange', () => {
74 | const userSession = new blockstack.UserSession();
75 | this.isAuthenticated = userSession.isUserSignedIn();
76 | });
77 | }
78 |
79 | async initCapacitor() {
80 | const { Device } = Plugins;
81 |
82 | const device = await Device.getInfo();
83 | if (device.platform !== 'web') {
84 | const { App, StatusBar } = Plugins;
85 | StatusBar.setBackgroundColor({ color: '#220631' });
86 |
87 | App.addListener('appUrlOpen', data => {
88 | if (data.url) {
89 | const authResponse = data.url.split(':')[1];
90 | if (authResponse) {
91 | window.location.href =
92 | window.location.href + '?authResponse=' + authResponse;
93 | }
94 | }
95 | });
96 | }
97 | }
98 |
99 | async handleSignOut() {
100 | // Clear all the users cache in localStorage
101 | CacheService.clear();
102 | // End users Blockstack session
103 | const userSession = new blockstack.UserSession();
104 | userSession.signUserOut();
105 |
106 | AnalyticsService.logEvent('logged-out');
107 | }
108 |
109 | render() {
110 | return [
111 |
112 |
113 |
114 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
122 |
123 |
124 | Menu
125 |
126 |
127 |
128 |
129 |
130 |
131 |
132 | Photos
133 |
134 |
135 |
136 |
137 |
138 | Albums
139 |
140 |
141 |
142 |
143 |
144 | Settings
145 |
146 |
147 |
148 | this.handleSignOut()}>
149 |
150 | Logout
151 |
152 |
153 |
154 |
155 |
156 |
157 |
158 |
159 |
160 |
161 |
162 |
163 |
164 |
165 | ];
166 | }
167 | }
168 |
--------------------------------------------------------------------------------
/src/pages/app-settings/app-settings.e2e.ts:
--------------------------------------------------------------------------------
1 | import { newE2EPage } from '@stencil/core/testing';
2 |
3 | describe('app-profile', () => {
4 | it('renders', async () => {
5 | const page = await newE2EPage();
6 | await page.setContent(' ');
7 |
8 | const element = await page.find('app-profile');
9 | expect(element).toHaveClass('hydrated');
10 | });
11 |
12 | it('displays the specified name', async () => {
13 | const page = await newE2EPage({ url: '/profile/joseph' });
14 |
15 | const element = await page.find('app-profile ion-content p');
16 | expect(element.textContent).toContain('My name is Joseph.');
17 | });
18 | });
19 |
--------------------------------------------------------------------------------
/src/pages/app-settings/app-settings.spec.ts:
--------------------------------------------------------------------------------
1 | import { AppProfile } from './app-profile';
2 |
3 | describe('app-profile', () => {
4 | it('builds', () => {
5 | expect(new AppProfile()).toBeTruthy();
6 | });
7 |
8 | describe('normalization', () => {
9 | it('returns a blank string if the name is undefined', () => {
10 | const component = new AppProfile();
11 | expect(component.formattedName()).toEqual('');
12 | });
13 |
14 | it('capitalizes the first letter', () => {
15 | const component = new AppProfile();
16 | component.name = 'quincy';
17 | expect(component.formattedName()).toEqual('Quincy');
18 | });
19 |
20 | it('lower-cases the following letters', () => {
21 | const component = new AppProfile();
22 | component.name = 'JOSEPH';
23 | expect(component.formattedName()).toEqual('Joseph');
24 | });
25 |
26 | it('handles single letter names', () => {
27 | const component = new AppProfile();
28 | component.name = 'q';
29 | expect(component.formattedName()).toEqual('Q');
30 | });
31 | });
32 | });
33 |
--------------------------------------------------------------------------------
/src/pages/app-settings/app-settings.tsx:
--------------------------------------------------------------------------------
1 | import { Component, State } from '@stencil/core';
2 | import PresentingService from '../../services/presenting-service';
3 | import AnalyticsService from '../../services/analytics-service';
4 | import SettingsService from '../../services/settings-service';
5 | import CacheService from '../../services/cache-service';
6 |
7 | @Component({
8 | tag: 'app-settings'
9 | })
10 | export class AppSettings {
11 | private present: PresentingService;
12 |
13 | @State() allowAnalytics: boolean;
14 |
15 | constructor() {
16 | this.present = new PresentingService();
17 | }
18 |
19 | async componentWillLoad() {
20 | this.allowAnalytics = await SettingsService.getAnalyticsSetting();
21 | AnalyticsService.logEvent('settings-page');
22 | }
23 |
24 | visitBlockstackProfile(event: any): void {
25 | if (event) {
26 | event.preventDefault();
27 | }
28 |
29 | this.present.openLink('https://browser.blockstack.org/profiles', '_blank');
30 |
31 | AnalyticsService.logEvent('blockstack-profile-link');
32 | }
33 |
34 | reportIssue(event: any): void {
35 | if (event) {
36 | event.preventDefault();
37 | }
38 |
39 | this.present.openLink(
40 | 'https://github.com/nerdic-coder/block-photos/issues/new',
41 | '_blank'
42 | );
43 |
44 | AnalyticsService.logEvent('report-issue-link');
45 | }
46 |
47 | donateCrypto(event: any): void {
48 | if (event) {
49 | event.preventDefault();
50 | }
51 |
52 | this.present.openLink(
53 | 'https://commerce.coinbase.com/checkout/9d35f08b-bd51-40b0-a502-b88250cffc6b',
54 | '_blank'
55 | );
56 |
57 | AnalyticsService.logEvent('donate-crypto-link');
58 | }
59 |
60 | upvoteProductHunt(event: any): void {
61 | if (event) {
62 | event.preventDefault();
63 | }
64 |
65 | this.present.openLink(
66 | 'https://www.producthunt.com/posts/block-photos',
67 | '_blank'
68 | );
69 |
70 | AnalyticsService.logEvent('product-hunt-link');
71 | }
72 |
73 | twitter(event: any): void {
74 | if (event) {
75 | event.preventDefault();
76 | }
77 |
78 | this.present.openLink('https://twitter.com/Block_Photos', '_blank');
79 |
80 | AnalyticsService.logEvent('twitter-link');
81 | }
82 |
83 | github(event: any): void {
84 | if (event) {
85 | event.preventDefault();
86 | }
87 |
88 | this.present.openLink(
89 | 'https://github.com/nerdic-coder/block-photos',
90 | '_blank'
91 | );
92 |
93 | AnalyticsService.logEvent('github-link');
94 | }
95 |
96 | sendEmail(event: any): void {
97 | if (event) {
98 | event.preventDefault();
99 | }
100 | this.present.openLink(
101 | 'mailto:johan@block-photos.com?subject=Block Photos Feedback'
102 | );
103 |
104 | AnalyticsService.logEvent('send-email-link');
105 | }
106 |
107 | changeAnalyticsSetting() {
108 | SettingsService.setAnalyticsSetting(!this.allowAnalytics);
109 | }
110 |
111 | async clearCache(event) {
112 | if (event) {
113 | event.preventDefault();
114 | }
115 | const actionSheetController = document.querySelector(
116 | 'ion-action-sheet-controller'
117 | );
118 | await actionSheetController.componentOnReady();
119 |
120 | const buttons = [
121 | {
122 | text: 'Clear cache',
123 | role: 'destructive',
124 | icon: 'reverse-camera',
125 | handler: () => {
126 | CacheService.clear();
127 | }
128 | }
129 | ];
130 |
131 | buttons.push({
132 | text: 'Cancel',
133 | icon: 'close',
134 | role: 'cancel',
135 | handler: null
136 | });
137 |
138 | const actionSheet = await actionSheetController.create({
139 | header: 'Are you sure you want to clear the cache?',
140 | buttons
141 | });
142 | await actionSheet.present();
143 | }
144 |
145 | render() {
146 | return [
147 |
148 |
149 | Settings
150 |
151 |
152 |
153 | Menu
154 |
155 |
156 |
157 |
158 | ,
159 |
160 |
161 |
162 |
163 | Share Analytics Data
164 | this.changeAnalyticsSetting()}
169 | />
170 |
171 |
172 |
173 | Help Block Photos improve the app by automatically share daily
174 | diagnostic and usage data. None of your photos or personal
175 | information will be tracked.
176 |
177 |
178 |
179 |
180 | this.clearCache(event)} detail={false}>
181 |
182 | Clear photos cache
183 |
184 |
185 |
186 | Photos starting to take up too much space? Empty the photos cache
187 | here to clear up some space on your device.
188 |
189 |
190 |
191 |
192 |
193 | this.visitBlockstackProfile(event)}
197 | >
198 |
199 | Go to profile on Blockstack
200 |
201 |
202 | this.sendEmail(event)}>
203 |
204 | Email johan@block-photos.com
205 |
206 |
207 | this.reportIssue(event)}>
208 |
209 | Report issue
210 |
211 |
212 |
213 | this.donateCrypto(event)}>
214 |
215 | Donate with Crypto
216 |
217 | this.upvoteProductHunt(event)}
221 | >
222 |
223 | Upvote on Product Hunt
224 |
225 | this.twitter(event)}>
226 |
227 | Follow us on Twitter
228 |
229 | this.github(event)}>
230 |
231 | Read our source code
232 |
233 |
234 | Block Photos - Version 4.2
235 |
236 | ];
237 | }
238 | }
239 |
--------------------------------------------------------------------------------
/src/pages/app-signin/app-signin.css:
--------------------------------------------------------------------------------
1 | .spinner-grid,
2 | .spinner-grid ion-row {
3 | height: 100%;
4 | max-height: 100%;
5 | }
6 |
7 | .signin ion-col {
8 | max-width: 450px;
9 | }
10 |
11 | .signin h1 {
12 | color: var(--light, #ffffff);
13 | }
14 |
15 | .signin ion-img {
16 | width: 70%;
17 | margin-left: auto;
18 | margin-right: auto;
19 | }
20 |
21 | .sc-ion-card-ios-h {
22 | margin: 16px 20px;
23 | }
24 |
--------------------------------------------------------------------------------
/src/pages/app-signin/app-signin.e2e.ts:
--------------------------------------------------------------------------------
1 | import { newE2EPage } from '@stencil/core/testing';
2 |
3 | describe('app-signin', () => {
4 | it('renders', async () => {
5 | const page = await newE2EPage();
6 | await page.setContent(' ');
7 |
8 | const element = await page.find('app-signin');
9 | expect(element).toHaveClass('hydrated');
10 | });
11 |
12 | it('contains a "Profile Page" button', async () => {
13 | const page = await newE2EPage();
14 | await page.setContent(' ');
15 |
16 | const element = await page.find('app-signin ion-content ion-button');
17 | expect(element.textContent).toEqual('Profile page');
18 | });
19 | });
20 |
--------------------------------------------------------------------------------
/src/pages/app-signin/app-signin.spec.ts:
--------------------------------------------------------------------------------
1 | import { AppSignin } from './app-signin';
2 |
3 | describe('app', () => {
4 | it('builds', () => {
5 | expect(new AppSignin()).toBeTruthy();
6 | });
7 | });
8 |
--------------------------------------------------------------------------------
/src/redirect.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Block Photos
5 |
15 |
16 |
42 |
43 |
44 |
45 |
--------------------------------------------------------------------------------
/src/services/albums-service.ts:
--------------------------------------------------------------------------------
1 | import uuidv4 from 'uuid/v4';
2 |
3 | import StorageService from './storage-service';
4 |
5 | export default class AlbumsService {
6 | static async getAlbums(updateCache?: boolean): Promise {
7 | let cachedAlbums = [];
8 | const errorsList = [];
9 | try {
10 | const rawCachedAlbums = await StorageService.getItem(
11 | 'albums-list.json',
12 | updateCache,
13 | updateCache
14 | );
15 | if (rawCachedAlbums) {
16 | cachedAlbums = JSON.parse(rawCachedAlbums);
17 | } else {
18 | errorsList.push('err_list');
19 | }
20 | } catch (error) {
21 | errorsList.push('err_list');
22 | }
23 |
24 | return {
25 | albums: cachedAlbums,
26 | errorsList
27 | };
28 | }
29 |
30 | static async createAlbum(albumName: string) {
31 | const albumsResponse = await AlbumsService.getAlbums(true);
32 | let albums = albumsResponse.albums;
33 | if ((!albums || albums == null) && albumsResponse.errorsList.length === 0) {
34 | albums = [];
35 | }
36 |
37 | const errorsList = [];
38 | const albumId = uuidv4();
39 | const metadata = {
40 | albumId,
41 | albumName,
42 | createdDate: new Date(),
43 | thumbnailId: null
44 | };
45 | try {
46 | await StorageService.setItem(albumId, '[]');
47 |
48 | albums.unshift(metadata);
49 | } catch (error) {
50 | errorsList.push({
51 | id: albumId,
52 | errorCode: 'err_failed'
53 | });
54 | }
55 |
56 | await StorageService.setItem('albums-list.json', JSON.stringify(albums));
57 | return { albums, errorsList };
58 | }
59 |
60 | static async updateAlbumName(
61 | albumId: string,
62 | albumName: string
63 | ): Promise {
64 | // id and metadata is required
65 | if (!albumId || !albumName) {
66 | return false;
67 | }
68 | const albumsResponse = await AlbumsService.getAlbums(true);
69 | const albums = albumsResponse.albums;
70 | let albumFound = false;
71 | let index = 0;
72 | for (const album of albums) {
73 | // Current album
74 | if (album.albumId === albumId) {
75 | albums[index].albumName = albumName;
76 | albumFound = true;
77 | break;
78 | }
79 | index++;
80 | }
81 |
82 | // Don't update if album don't exist
83 | if (!albumFound) {
84 | return false;
85 | }
86 |
87 | await StorageService.setItem('albums-list.json', JSON.stringify(albums));
88 |
89 | return albums;
90 | }
91 |
92 | static async updateAlbumThumbnail(
93 | albumId: string,
94 | thumbnailId: string
95 | ): Promise {
96 | // id and metadata is required
97 | if (!albumId || !thumbnailId) {
98 | return false;
99 | }
100 | const albumsResponse = await AlbumsService.getAlbums(true);
101 | const albums = albumsResponse.albums;
102 | let albumFound = false;
103 | let index = 0;
104 | for (const album of albums) {
105 | // Current album
106 | if (album.albumId === albumId) {
107 | albums[index].thumbnailId = thumbnailId;
108 | albumFound = true;
109 | break;
110 | }
111 | index++;
112 | }
113 |
114 | // Don't update if album don't exist
115 | if (!albumFound) {
116 | return false;
117 | }
118 |
119 | await StorageService.setItem('albums-list.json', JSON.stringify(albums));
120 |
121 | return albums;
122 | }
123 |
124 | static async deleteAlbum(albumId: string): Promise {
125 | let returnState = false;
126 | try {
127 | StorageService.deleteItem(albumId);
128 | returnState = true;
129 | } catch (error) {
130 | returnState = false;
131 | }
132 |
133 | if (!returnState) {
134 | return false;
135 | }
136 |
137 | const albumResponse = await AlbumsService.getAlbums(true);
138 | const albums = albumResponse.albums;
139 |
140 | let index = 0;
141 | for (const photo of albums) {
142 | if (albumId === photo.albumId) {
143 | albums.splice(index, 1);
144 | await StorageService.setItem(
145 | 'albums-list.json',
146 | JSON.stringify(albums)
147 | );
148 | return albums;
149 | }
150 | index++;
151 | }
152 | return false;
153 | }
154 |
155 | static async getAlbumMetaData(albumId: string): Promise {
156 | let response = {};
157 | const albumsResponse = await AlbumsService.getAlbums();
158 | const albums = albumsResponse.albums;
159 |
160 | let index = 0;
161 | for (const album of albums) {
162 | // Current album
163 | if (album.albumId === albumId) {
164 | response = albums[index];
165 | break;
166 | }
167 | index++;
168 | }
169 |
170 | return response;
171 | }
172 | }
173 |
--------------------------------------------------------------------------------
/src/services/analytics-service.ts:
--------------------------------------------------------------------------------
1 | import SettingsService from './settings-service';
2 |
3 | declare var amplitude;
4 |
5 | export default class AnalyticsService {
6 | static async logEvent(event: string): Promise {
7 | if (amplitude && (await SettingsService.getAnalyticsSetting())) {
8 | amplitude.getInstance().logEvent(event);
9 | }
10 | }
11 | }
12 |
--------------------------------------------------------------------------------
/src/services/cache-service.spec.ts:
--------------------------------------------------------------------------------
1 | import CacheService from './cache-service';
2 | import localForage from 'localforage';
3 |
4 | jest.mock('localforage');
5 |
6 | describe('CacheService Test Suites', () => {
7 | it('initialize without crashing', () => {
8 | new CacheService();
9 | });
10 |
11 | it('get photos list', async () => {
12 | const mockResponse = [
13 | {
14 | id: 'test123.jpg',
15 | uploadedDate: '2018-07-11T21:58:39.754Z'
16 | }
17 | ];
18 |
19 | localForage.getItem.mockReturnValue(
20 | Promise.resolve(JSON.stringify(mockResponse))
21 | );
22 |
23 | const cacheService = new CacheService();
24 | const response = await cacheService.getItem('picture-list.json');
25 |
26 | const jsonResponse = JSON.parse(response);
27 | expect(Array.isArray(jsonResponse)).toBe(true);
28 | expect(jsonResponse).toEqual(mockResponse);
29 | });
30 |
31 | it('get empty photos list', async () => {
32 | const mockResponse = [];
33 |
34 | localForage.getItem.mockReturnValue(
35 | Promise.resolve(JSON.stringify(mockResponse))
36 | );
37 |
38 | const cacheService = new CacheService();
39 | const response = await cacheService.getItem('picture-list.json');
40 |
41 | const jsonResponse = JSON.parse(response);
42 | expect(Array.isArray(jsonResponse)).toBe(true);
43 | expect(jsonResponse).toEqual(mockResponse);
44 | });
45 |
46 | it('get null photos list', async () => {
47 | const mockResponse = null;
48 |
49 | localForage.getItem.mockReturnValue(
50 | Promise.resolve(JSON.stringify(mockResponse))
51 | );
52 |
53 | const cacheService = new CacheService();
54 | const response = await cacheService.getItem('picture-list.json');
55 |
56 | const jsonResponse = JSON.parse(response);
57 | expect(Array.isArray(jsonResponse)).toBe(false);
58 | expect(jsonResponse).toBe(null);
59 | });
60 |
61 | it('cache photo', async () => {
62 | const cacheService = new CacheService();
63 | await cacheService.setItem('test1.png', [
64 | {
65 | filename: 'test1.png',
66 | data: 'trash'
67 | }
68 | ]);
69 |
70 | // const response = await cacheService.getItem('test1.png');
71 | });
72 | });
73 |
--------------------------------------------------------------------------------
/src/services/cache-service.ts:
--------------------------------------------------------------------------------
1 | import localForage from 'localforage';
2 |
3 | export default class CacheService {
4 | static async getItem(itemId: string): Promise {
5 | return localForage.getItem(itemId);
6 | }
7 |
8 | static async setItem(itemId: string, itemValue: any): Promise {
9 | return localForage.setItem(itemId, itemValue);
10 | }
11 |
12 | static async deleteItem(itemId: string): Promise {
13 | return localForage.removeItem(itemId);
14 | }
15 |
16 | static async clear(): Promise {
17 | return localForage.clear();
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/services/presenting-service.ts:
--------------------------------------------------------------------------------
1 | import PhotosService from './photos-service';
2 | import isElectron from 'is-electron';
3 |
4 | export default class PresentingService {
5 | private loadingElement: HTMLIonLoadingElement;
6 |
7 | async loading(message: string, duration?: number): Promise {
8 | const loadingController = document.querySelector('ion-loading-controller');
9 | await loadingController.componentOnReady();
10 |
11 | this.loadingElement = await loadingController.create({
12 | message,
13 | spinner: 'circles',
14 | duration
15 | });
16 | return this.loadingElement.present();
17 | }
18 |
19 | async dismissLoading(): Promise {
20 | if (this.loadingElement) {
21 | await this.loadingElement.dismiss();
22 | }
23 | }
24 |
25 | async presentToolbarLoader(message: string): Promise {
26 | const loadingToolbar = document.querySelector('.loadingToolbar');
27 |
28 | if (loadingToolbar) {
29 | const loadingTitle = loadingToolbar.querySelector('ion-title');
30 | await loadingTitle.componentOnReady();
31 | loadingTitle.innerText = message;
32 | loadingToolbar.classList.add('show');
33 | }
34 | }
35 |
36 | toolbarLoaderIsPresent(): boolean {
37 | const loadingToolbar = document.querySelector('.loadingToolbar');
38 |
39 | if (loadingToolbar) {
40 | return loadingToolbar.classList.contains('show');
41 | } else {
42 | return false;
43 | }
44 | }
45 |
46 | async dismissToolbarLoader(): Promise {
47 | const loadingToolbar = document.querySelector('.loadingToolbar');
48 |
49 | if (loadingToolbar) {
50 | const loadingTitle = loadingToolbar.querySelector('ion-title');
51 | await loadingTitle.componentOnReady();
52 | loadingTitle.innerText = '';
53 | loadingToolbar.classList.remove('show');
54 | }
55 | }
56 |
57 | async toast(message: string): Promise {
58 | const toastController = document.querySelector('ion-toast-controller');
59 | await toastController.componentOnReady();
60 |
61 | const toast = await toastController.create({
62 | message,
63 | showCloseButton: true,
64 | color: 'primary'
65 | });
66 | return toast.present();
67 | }
68 |
69 | async deletePhotos(
70 | ids: string[],
71 | endCallback: any,
72 | albumId?: string,
73 | startCallback?: any
74 | ): Promise {
75 | if (!ids || ids.length < 1) {
76 | return;
77 | }
78 |
79 | let header = 'Delete ' + ids.length + ' photos?';
80 | if (ids.length === 1) {
81 | header = 'Delete ' + ids.length + ' photo?';
82 | }
83 | const actionSheetController = document.querySelector(
84 | 'ion-action-sheet-controller'
85 | );
86 | await actionSheetController.componentOnReady();
87 |
88 | const buttons = [
89 | {
90 | text: 'Delete from app',
91 | role: 'destructive',
92 | icon: 'trash',
93 | handler: () => {
94 | if (startCallback) {
95 | startCallback();
96 | }
97 |
98 | PhotosService.deletePhotos(ids).then(async result => {
99 | if (result === true) {
100 | endCallback();
101 | } else {
102 | this.errorAlert(
103 | 'Removal failed',
104 | 'The removal of some photos failed. Please try again in a few minutes!'
105 | );
106 | endCallback();
107 | }
108 | });
109 | }
110 | }
111 | ];
112 |
113 | if (albumId) {
114 | buttons.push({
115 | text: 'Remove from album',
116 | role: 'destructive',
117 | icon: 'remove-circle',
118 | handler: () => {
119 | if (startCallback) {
120 | startCallback();
121 | }
122 | PhotosService.removePhotosFromList(ids, albumId).then(
123 | async result => {
124 | if (result === true) {
125 | endCallback();
126 | } else {
127 | this.errorAlert(
128 | 'Removal failed',
129 | 'The removal of some photos failed. Please try again in a few minutes!'
130 | );
131 | endCallback();
132 | }
133 | }
134 | );
135 | }
136 | });
137 | }
138 |
139 | buttons.push({
140 | text: 'Cancel',
141 | icon: 'close',
142 | role: 'cancel',
143 | handler: null
144 | });
145 |
146 | const actionSheet = await actionSheetController.create({
147 | header,
148 | buttons
149 | });
150 | await actionSheet.present();
151 | }
152 |
153 | async errorAlert(header: string, message: string): Promise {
154 | const alertController = document.querySelector('ion-alert-controller');
155 | await alertController.componentOnReady();
156 |
157 | const alert = await alertController.create({
158 | header,
159 | subHeader: '',
160 | message,
161 | buttons: ['OK']
162 | });
163 | return alert.present();
164 | }
165 |
166 | openLink(url: string, target?: string): void {
167 | if (isElectron()) {
168 | const electron = window['require']('electron');
169 | electron.shell.openExternal(url);
170 | } else {
171 | window.open(url, target);
172 | }
173 | }
174 | }
175 |
--------------------------------------------------------------------------------
/src/services/settings-service.ts:
--------------------------------------------------------------------------------
1 | import StorageService from './storage-service';
2 |
3 | export default class SettingsService {
4 | static async getAnalyticsSetting(updateCache?: boolean): Promise {
5 | return (
6 | (await StorageService.getItem('analytics-settings', updateCache)) ===
7 | 'true'
8 | );
9 | }
10 |
11 | static async setAnalyticsSetting(allowAnalytics: boolean): Promise {
12 | return StorageService.setItem(
13 | 'analytics-settings',
14 | allowAnalytics.toString()
15 | );
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/src/services/storage-service.ts:
--------------------------------------------------------------------------------
1 | import CacheService from './cache-service';
2 |
3 | declare var blockstack;
4 |
5 | export default class StorageService {
6 | static async getItem(
7 | itemId: string,
8 | updateCache?: boolean,
9 | forceUpdateCache?: boolean
10 | ) {
11 | let item = await CacheService.getItem(itemId);
12 | if (!item || forceUpdateCache) {
13 | const userSession = new blockstack.UserSession();
14 | item = await userSession.getFile(itemId);
15 | if (updateCache || forceUpdateCache) {
16 | CacheService.setItem(itemId, item);
17 | }
18 | }
19 |
20 | return item;
21 | }
22 |
23 | static async setItem(
24 | itemId: string,
25 | itemValue: any,
26 | cacheItem = true
27 | ): Promise {
28 | const userSession = new blockstack.UserSession();
29 | await userSession.putFile(itemId, itemValue);
30 | if (cacheItem) {
31 | await CacheService.setItem(itemId, itemValue);
32 | }
33 |
34 | const timeStamp = Math.floor(Date.now() / 1000);
35 | userSession.putFile('block-photos-last-updated', timeStamp.toString());
36 | CacheService.setItem('block-photos-last-checked', timeStamp.toString());
37 | }
38 |
39 | static async deleteItem(itemId: string) {
40 | const userSession = new blockstack.UserSession();
41 | await userSession.deleteFile(itemId);
42 | await CacheService.deleteItem(itemId);
43 | }
44 |
45 | static clear() {
46 | CacheService.clear();
47 | }
48 |
49 | static async checkUpdatedTimestamp(): Promise {
50 | const checkedTimestamp = await CacheService.getItem(
51 | 'block-photos-last-checked'
52 | );
53 | const timeStamp = Math.floor(Date.now() / 1000);
54 | if (!checkedTimestamp) {
55 | await CacheService.setItem(
56 | 'block-photos-last-checked',
57 | timeStamp.toString()
58 | );
59 | return false;
60 | } else {
61 | const userSession = new blockstack.UserSession();
62 | const updatedTimeStamp = await userSession.getFile(
63 | 'block-photos-last-updated'
64 | );
65 | if (updatedTimeStamp > checkedTimestamp) {
66 | await CacheService.setItem(
67 | 'block-photos-last-checked',
68 | timeStamp.toString()
69 | );
70 | return true;
71 | } else {
72 | return false;
73 | }
74 | }
75 | }
76 | }
77 |
--------------------------------------------------------------------------------
/src/services/upload-service.ts:
--------------------------------------------------------------------------------
1 | import loadImage from 'blueimp-load-image';
2 | import uuidv4 from 'uuid/v4';
3 | import PhotosService from './photos-service';
4 | import PresentingService from './presenting-service';
5 | import { PhotoType } from '../models/photo-type';
6 | import { Plugins } from '@capacitor/core';
7 |
8 | export default class UploadService {
9 | private root: any;
10 | private present: PresentingService;
11 | private callback: any;
12 | private startedCallback: any;
13 | private dropEventBinding: any;
14 | private handleFileSelectEventBinding: any;
15 | private albumId: string;
16 |
17 | constructor(callback: any, albumId?: string, startedCallback?: any) {
18 | this.present = new PresentingService();
19 | this.callback = callback;
20 | this.startedCallback = startedCallback;
21 | this.dropEventBinding = this.dropEvent.bind(this);
22 | this.handleFileSelectEventBinding = this.handleFileSelectEvent.bind(this);
23 | this.root = document.getElementById('photos-list');
24 | this.albumId = albumId;
25 | }
26 |
27 | addEventListeners(fileDialog: boolean): void {
28 | this.root = document.getElementById('photos-list');
29 | if (this.root) {
30 | this.root.addEventListener('dragover', this.dragoverEvent);
31 | this.root.addEventListener('drop', this.dropEventBinding);
32 | }
33 | if (fileDialog && document.getElementById('file-upload')) {
34 | document
35 | .getElementById('file-upload')
36 | .addEventListener('change', this.handleFileSelectEventBinding, false);
37 | }
38 | }
39 |
40 | removeEventListeners(fileDialog: boolean): void {
41 | if (this.root) {
42 | this.root.removeEventListener('dragover', this.dragoverEvent);
43 | this.root.removeEventListener('drop', this.dropEventBinding);
44 | }
45 | if (fileDialog && document.getElementById('file-upload')) {
46 | document
47 | .getElementById('file-upload')
48 | .removeEventListener('change', this.handleFileSelectEventBinding);
49 | }
50 | }
51 |
52 | dragoverEvent(event: any): void {
53 | event.stopPropagation();
54 | event.preventDefault();
55 | event.dataTransfer.dropEffect = 'copy';
56 | }
57 |
58 | dropEvent(event: any): void {
59 | event.stopPropagation();
60 | event.preventDefault();
61 |
62 | if (event.dataTransfer.items) {
63 | const photosToUpload = [];
64 | for (let i = 0; i < event.dataTransfer.items.length; i++) {
65 | const item = {
66 | kind: event.dataTransfer.items[i].kind,
67 | file: event.dataTransfer.items[i].getAsFile()
68 | };
69 | photosToUpload.push(item);
70 | }
71 | this.processUpload(photosToUpload, 0);
72 |
73 | event.dataTransfer.items.clear();
74 | }
75 | }
76 |
77 | handleFileSelectEvent(event: any): void {
78 | event.stopPropagation();
79 | event.preventDefault();
80 |
81 | const files = event.target.files;
82 |
83 | if (files) {
84 | const photosToUpload = [];
85 | for (let i = 0; i < files.length; i++) {
86 | const item = {
87 | kind: 'file',
88 | file: files[i]
89 | };
90 | photosToUpload.push(item);
91 | }
92 | this.processUpload(photosToUpload, 0);
93 | }
94 | }
95 |
96 | async processUpload(list: any, currentIndex: number): Promise {
97 | if (currentIndex === 0) {
98 | if (this.startedCallback && typeof this.startedCallback === 'function') {
99 | // execute the callback, passing parameters as necessary
100 | this.startedCallback();
101 | }
102 | }
103 | this.present.presentToolbarLoader(
104 | 'Uploading photo ' + (currentIndex + 1) + ' of ' + list.length + '.'
105 | );
106 | // If dropped items aren't files, reject them
107 | if (list[currentIndex]) {
108 | const tempFile = list[currentIndex].file;
109 | if (list[currentIndex].kind === 'file') {
110 | if (tempFile.type.indexOf('image') !== -1) {
111 | const { Device } = Plugins;
112 | const info = await Device.getInfo();
113 | let thumbnailData = null;
114 | let viewerData = null;
115 | thumbnailData = await PhotosService.compressPhoto(
116 | tempFile,
117 | PhotoType.Thumbnail,
118 | tempFile.type
119 | );
120 | if (info.model !== 'iPhone' && info.model !== 'iPad') {
121 | viewerData = await PhotosService.compressPhoto(
122 | tempFile,
123 | PhotoType.Viewer,
124 | tempFile.type
125 | );
126 | }
127 | loadImage.parseMetaData(
128 | tempFile,
129 | data => {
130 | const reader = new FileReader();
131 |
132 | let orientation = 1;
133 | if (data && data.exif) {
134 | orientation = data.exif.get('Orientation');
135 | }
136 |
137 | // Closure to capture the file information.
138 | reader.onload = ((loadedFile, loadedList, loadedOrientation) => {
139 | return async event => {
140 | const originalData = event.target.result;
141 | if (loadedOrientation) {
142 | loadedFile.exifdata = {
143 | tags: {
144 | Orientation: loadedOrientation,
145 | OriginalOrientation: loadedOrientation
146 | }
147 | };
148 | }
149 | const photoId: string =
150 | uuidv4() +
151 | loadedFile.name.replace('.', '').replace(' ', '');
152 | const metadata: PhotoMetadata = {
153 | id: photoId,
154 | filename: loadedFile.name,
155 | stats: loadedFile,
156 | type: tempFile.type,
157 | size: tempFile.size,
158 | uploadedDate: new Date(),
159 | albums: [this.albumId]
160 | };
161 |
162 | await this.uploadPhoto(
163 | metadata,
164 | originalData,
165 | thumbnailData,
166 | viewerData
167 | );
168 | if (loadedList[currentIndex + 1]) {
169 | this.processUpload(loadedList, currentIndex + 1);
170 | } else {
171 | this.uploadFilesDone();
172 | }
173 | };
174 | })(tempFile, list, orientation);
175 | // Read in the image file as a data URL.
176 | reader.readAsDataURL(tempFile);
177 | },
178 | {
179 | maxMetaDataSize: 262144,
180 | disableImageHead: false
181 | }
182 | );
183 | } else {
184 | this.present.toast(
185 | 'The file "' +
186 | tempFile.name +
187 | '" could not be uploaded, are you sure it\'s a photo?'
188 | );
189 | if (list[currentIndex + 1]) {
190 | this.processUpload(list, currentIndex + 1);
191 | } else {
192 | this.uploadFilesDone();
193 | }
194 | }
195 | } else {
196 | if (tempFile && tempFile.name) {
197 | this.present.toast(
198 | 'The file "' +
199 | tempFile.name +
200 | '" could not be uploaded, are you sure it\'s a photo?'
201 | );
202 | } else {
203 | this.present.toast(
204 | "One of the files could not be uploaded, are you sure it's a photo?"
205 | );
206 | }
207 | if (list[currentIndex + 1]) {
208 | this.processUpload(list, currentIndex + 1);
209 | } else {
210 | this.uploadFilesDone();
211 | }
212 | }
213 | } else {
214 | this.present.toast(
215 | "The file could not be uploaded, are you sure it's a photo?"
216 | );
217 | if (list[currentIndex + 1]) {
218 | this.processUpload(list, currentIndex + 1);
219 | } else {
220 | this.uploadFilesDone();
221 | }
222 | }
223 | }
224 |
225 | async uploadPhoto(
226 | metadata: PhotoMetadata,
227 | data: any,
228 | thumbnailData?: any,
229 | viewerData?: any
230 | ): Promise {
231 | if (metadata && data) {
232 | const response = await PhotosService.uploadPhoto(
233 | metadata,
234 | data,
235 | this.albumId,
236 | thumbnailData,
237 | viewerData
238 | );
239 | if (response.errorsList && response.errorsList.length > 0) {
240 | for (const error of response.errorsList) {
241 | if (error.errorCode === 'err_filesize') {
242 | this.present.toast(
243 | 'Failed to upload "' +
244 | error.id +
245 | '", photo exceeds file size limit of 5MB.'
246 | );
247 | } else {
248 | this.present.toast('Failed to upload "' + error.id + '".');
249 | }
250 | }
251 | }
252 | } else {
253 | this.present.toast('Failed to upload "' + metadata.filename + '".');
254 | }
255 | }
256 |
257 | uploadFilesDone(): void {
258 | this.present.dismissToolbarLoader();
259 | if (this.callback && typeof this.callback === 'function') {
260 | // execute the callback, passing parameters as necessary
261 | this.callback();
262 | }
263 | }
264 | }
265 |
--------------------------------------------------------------------------------
/src/site.webmanifest:
--------------------------------------------------------------------------------
1 | {
2 | "name": "Block Photos",
3 | "short_name": "Block Photos",
4 | "icons": [
5 | {
6 | "src": "/android-chrome-36x36.png",
7 | "sizes": "36x36",
8 | "type": "image/png"
9 | },
10 | {
11 | "src": "/android-chrome-48x48.png",
12 | "sizes": "48x48",
13 | "type": "image/png"
14 | },
15 | {
16 | "src": "/android-chrome-72x72.png",
17 | "sizes": "72x72",
18 | "type": "image/png"
19 | },
20 | {
21 | "src": "/android-chrome-96x96.png",
22 | "sizes": "96x96",
23 | "type": "image/png"
24 | },
25 | {
26 | "src": "/android-chrome-144x144.png",
27 | "sizes": "144x144",
28 | "type": "image/png"
29 | },
30 | {
31 | "src": "/android-chrome-192x192.png",
32 | "sizes": "192x192",
33 | "type": "image/png"
34 | },
35 | {
36 | "src": "/android-chrome-256x256.png",
37 | "sizes": "256x256",
38 | "type": "image/png"
39 | },
40 | {
41 | "src": "/android-chrome-384x384.png",
42 | "sizes": "384x384",
43 | "type": "image/png"
44 | },
45 | {
46 | "src": "/android-chrome-512x512.png",
47 | "sizes": "512x512",
48 | "type": "image/png"
49 | }
50 | ],
51 | "theme_color": "#220631",
52 | "background_color": "#ffffff",
53 | "start_url": "https://app.block-photos.com/",
54 | "display": "standalone"
55 | }
56 |
--------------------------------------------------------------------------------
/src/sw.js:
--------------------------------------------------------------------------------
1 | importScripts('workbox-v3.4.1/workbox-sw.js');
2 |
3 | /*
4 | This is our code to handle push events.
5 | */
6 | self.addEventListener('push', event => {
7 | console.log('[Service Worker] Push Received.');
8 | console.log(`[Service Worker] Push had this data: "${event.data.text()}"`);
9 |
10 | const title = 'Push Notification';
11 | const options = {
12 | body: `${event.data.text()}`,
13 | icon: 'images/icon.png',
14 | badge: 'images/badge.png'
15 | };
16 |
17 | event.waitUntil(self.registration.showNotification(title, options));
18 | });
19 |
20 | self.workbox.precaching.precacheAndRoute([]);
21 |
--------------------------------------------------------------------------------
/stencil.config.ts:
--------------------------------------------------------------------------------
1 | import { Config } from '@stencil/core';
2 | import builtins from 'rollup-plugin-node-builtins';
3 | import globals from 'rollup-plugin-node-globals';
4 | // https://stenciljs.com/docs/config
5 |
6 | export const config: Config = {
7 | outputTargets: [
8 | {
9 | type: 'www',
10 | serviceWorker: {
11 | swSrc: 'src/sw.js'
12 | }
13 | }
14 | ],
15 | copy: [
16 | { src: '_headers' },
17 | { src: '_redirects' },
18 | { src: 'browserconfig.xml' },
19 | { src: 'redirect.html' },
20 | { src: 'site.webmanifest' }
21 | ],
22 | globalScript: 'src/global/app.ts',
23 | globalStyle: 'src/global/app.css',
24 | nodeResolve: {
25 | browser: true,
26 | preferBuiltins: false
27 | },
28 | plugins: [builtins(), globals()]
29 | };
30 |
--------------------------------------------------------------------------------
/tests/Signin.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | 'Signin' : function (browser) {
3 | browser
4 | .url('http://localhost:9876')
5 | .waitForElementVisible('body', 1000)
6 | // .setValue('input[type=text]', 'nightwatch')
7 | // .waitForElementVisible('button[name=btnG]', 1000)
8 | // .click('button[name=btnG]')
9 | .assert.containsText('ion-title', 'Block Photos')
10 | .assert.containsText('h1', 'Welcome to Block Photos!')
11 | .assert.containsText('ion-button', 'SIGN IN WITH BLOCKSTACK')
12 | .click('ion-button')
13 | // .assert.containsText('#main', 'Night Watch')
14 | .end();
15 | }
16 | };
17 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "allowSyntheticDefaultImports": true,
4 | "allowUnreachableCode": false,
5 | "declaration": false,
6 | "experimentalDecorators": true,
7 | "lib": [
8 | "dom",
9 | "es2017"
10 | ],
11 | "moduleResolution": "node",
12 | "module": "esnext",
13 | "target": "es2017",
14 | "noUnusedLocals": true,
15 | "noUnusedParameters": true,
16 | "jsx": "react",
17 | "jsxFactory": "h",
18 | "resolveJsonModule": true,
19 | "suppressImplicitAnyIndexErrors": true
20 | },
21 | "include": [
22 | "src"
23 | ],
24 | "exclude": [
25 | "node_modules"
26 | ]
27 | }
28 |
--------------------------------------------------------------------------------
/tslint.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "tslint-ionic-rules",
3 | "rules": {
4 | "no-import-side-effect": false
5 | }
6 | }
7 |
--------------------------------------------------------------------------------
/webpack.config.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 |
3 | // copy manifest.json to the path: 'public/build'
4 | // this will allow for the authRequest to see the file at www.example.com/manifest.json
5 | const CopyWebpackPlugin = require('copy-webpack-plugin');
6 | const HeadersPlugin = new CopyWebpackPlugin([{ from: 'public/', to: '.' }]);
7 | const IonicDistPlugin = new CopyWebpackPlugin([ { from: 'node_modules/@ionic/core/dist', to: 'ionic' } ]);
8 | const WorkboxPlugin = require('workbox-webpack-plugin');
9 |
10 | const HtmlWebpackPlugin = require('html-webpack-plugin');
11 |
12 | let PUBLIC_PATH = './';
13 |
14 | process.argv.forEach(function (val) {
15 | if (val.includes('target=web')) {
16 | PUBLIC_PATH = '/';
17 | }
18 | });
19 |
20 | const HtmlWebpackPluginConfig = new HtmlWebpackPlugin({
21 | template: './public/index.html',
22 | filename: 'index.html',
23 | inject: 'body',
24 | base: PUBLIC_PATH
25 | });
26 |
27 | module.exports = {
28 | entry: ['@babel/polyfill', './src/index.js'],
29 | target: 'electron-main',
30 | output: {
31 | path: path.resolve('www'),
32 | filename: '[name].bundle.js',
33 | chunkFilename: '[name].bundle.js',
34 | publicPath: PUBLIC_PATH
35 | },
36 | devServer: {
37 | https: false,
38 | historyApiFallback: {
39 | disableDotRule: true
40 | },
41 | watchOptions: { aggregateTimeout: 300, poll: 1000 },
42 | headers: {
43 | "Access-Control-Allow-Origin": "*",
44 | "Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, PATCH, OPTIONS",
45 | "Access-Control-Allow-Headers": "X-Requested-With, content-type, Authorization",
46 | },
47 | port: 9876
48 | },
49 | module: {
50 | rules: [
51 | {
52 | test: /\.js$/,
53 | loader: 'babel-loader',
54 | exclude: /\.(\/node_modules\/|\/build\/|\/electron\/|\/src\/server.js)(\?\S*)?$/
55 | },
56 | { test: /\.jsx$/, loader: 'babel-loader', exclude: /node_modules/ },
57 | {
58 | test: /\.(eot|woff|woff2|ttf|svg|png|jpe?g|gif)(\?\S*)?$/,
59 | loader: 'file-loader!url-loader',
60 | },
61 | { test: /\.css$/, loader: 'style-loader!css-loader' }
62 | ]
63 | },
64 | plugins: [HtmlWebpackPluginConfig, IonicDistPlugin, HeadersPlugin,
65 | new WorkboxPlugin.InjectManifest({
66 | swSrc: './public/service-worker.js',
67 | })
68 | ]
69 | }
70 |
--------------------------------------------------------------------------------